Files
2025-yatcpu/lab1/实验报告/实验报告:实验一 单周期 RISC-V CPU 设计与实现.html
2025-10-10 20:07:22 +08:00

273 lines
37 KiB
HTML
Raw Blame History

This file contains ambiguous Unicode characters

This file contains Unicode characters that might be confused with other characters. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.

<!DOCTYPE html><html><head>
<title>实验报告:实验一 单周期 RISC-V CPU 设计与实现</title>
<meta charset="utf-8">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<link rel="stylesheet" href="file:///c:\Users\26586\.vscode\extensions\shd101wyy.markdown-preview-enhanced-0.8.19\crossnote\dependencies\katex\katex.min.css">
<style>
code[class*=language-],pre[class*=language-]{color:#333;background:0 0;font-family:Consolas,"Liberation Mono",Menlo,Courier,monospace;text-align:left;white-space:pre;word-spacing:normal;word-break:normal;word-wrap:normal;line-height:1.4;-moz-tab-size:8;-o-tab-size:8;tab-size:8;-webkit-hyphens:none;-moz-hyphens:none;-ms-hyphens:none;hyphens:none}pre[class*=language-]{padding:.8em;overflow:auto;border-radius:3px;background:#f5f5f5}:not(pre)>code[class*=language-]{padding:.1em;border-radius:.3em;white-space:normal;background:#f5f5f5}.token.blockquote,.token.comment{color:#969896}.token.cdata{color:#183691}.token.doctype,.token.macro.property,.token.punctuation,.token.variable{color:#333}.token.builtin,.token.important,.token.keyword,.token.operator,.token.rule{color:#a71d5d}.token.attr-value,.token.regex,.token.string,.token.url{color:#183691}.token.atrule,.token.boolean,.token.code,.token.command,.token.constant,.token.entity,.token.number,.token.property,.token.symbol{color:#0086b3}.token.prolog,.token.selector,.token.tag{color:#63a35c}.token.attr-name,.token.class,.token.class-name,.token.function,.token.id,.token.namespace,.token.pseudo-class,.token.pseudo-element,.token.url-reference .token.variable{color:#795da3}.token.entity{cursor:help}.token.title,.token.title .token.punctuation{font-weight:700;color:#1d3e81}.token.list{color:#ed6a43}.token.inserted{background-color:#eaffea;color:#55a532}.token.deleted{background-color:#ffecec;color:#bd2c00}.token.bold{font-weight:700}.token.italic{font-style:italic}.language-json .token.property{color:#183691}.language-markup .token.tag .token.punctuation{color:#333}.language-css .token.function,code.language-css{color:#0086b3}.language-yaml .token.atrule{color:#63a35c}code.language-yaml{color:#183691}.language-ruby .token.function{color:#333}.language-markdown .token.url{color:#795da3}.language-makefile .token.symbol{color:#795da3}.language-makefile .token.variable{color:#183691}.language-makefile .token.builtin{color:#0086b3}.language-bash .token.keyword{color:#0086b3}pre[data-line]{position:relative;padding:1em 0 1em 3em}pre[data-line] .line-highlight-wrapper{position:absolute;top:0;left:0;background-color:transparent;display:block;width:100%}pre[data-line] .line-highlight{position:absolute;left:0;right:0;padding:inherit 0;margin-top:1em;background:hsla(24,20%,50%,.08);background:linear-gradient(to right,hsla(24,20%,50%,.1) 70%,hsla(24,20%,50%,0));pointer-events:none;line-height:inherit;white-space:pre}pre[data-line] .line-highlight:before,pre[data-line] .line-highlight[data-end]:after{content:attr(data-start);position:absolute;top:.4em;left:.6em;min-width:1em;padding:0 .5em;background-color:hsla(24,20%,50%,.4);color:#f4f1ef;font:bold 65%/1.5 sans-serif;text-align:center;vertical-align:.3em;border-radius:999px;text-shadow:none;box-shadow:0 1px #fff}pre[data-line] .line-highlight[data-end]:after{content:attr(data-end);top:auto;bottom:.4em}html body{font-family:'Helvetica Neue',Helvetica,'Segoe UI',Arial,freesans,sans-serif;font-size:16px;line-height:1.6;color:#333;background-color:#fff;overflow:initial;box-sizing:border-box;word-wrap:break-word}html body>:first-child{margin-top:0}html body h1,html body h2,html body h3,html body h4,html body h5,html body h6{line-height:1.2;margin-top:1em;margin-bottom:16px;color:#000}html body h1{font-size:2.25em;font-weight:300;padding-bottom:.3em}html body h2{font-size:1.75em;font-weight:400;padding-bottom:.3em}html body h3{font-size:1.5em;font-weight:500}html body h4{font-size:1.25em;font-weight:600}html body h5{font-size:1.1em;font-weight:600}html body h6{font-size:1em;font-weight:600}html body h1,html body h2,html body h3,html body h4,html body h5{font-weight:600}html body h5{font-size:1em}html body h6{color:#5c5c5c}html body strong{color:#000}html body del{color:#5c5c5c}html body a:not([href]){color:inherit;text-decoration:none}html body a{color:#08c;text-decoration:none}html body a:hover{color:#00a3f5;text-decoration:none}html body img{max-width:100%}html body>p{margin-top:0;margin-bottom:16px;word-wrap:break-word}html body>ol,html body>ul{margin-bottom:16px}html body ol,html body ul{padding-left:2em}html body ol.no-list,html body ul.no-list{padding:0;list-style-type:none}html body ol ol,html body ol ul,html body ul ol,html body ul ul{margin-top:0;margin-bottom:0}html body li{margin-bottom:0}html body li.task-list-item{list-style:none}html body li>p{margin-top:0;margin-bottom:0}html body .task-list-item-checkbox{margin:0 .2em .25em -1.8em;vertical-align:middle}html body .task-list-item-checkbox:hover{cursor:pointer}html body blockquote{margin:16px 0;font-size:inherit;padding:0 15px;color:#5c5c5c;background-color:#f0f0f0;border-left:4px solid #d6d6d6}html body blockquote>:first-child{margin-top:0}html body blockquote>:last-child{margin-bottom:0}html body hr{height:4px;margin:32px 0;background-color:#d6d6d6;border:0 none}html body table{margin:10px 0 15px 0;border-collapse:collapse;border-spacing:0;display:block;width:100%;overflow:auto;word-break:normal;word-break:keep-all}html body table th{font-weight:700;color:#000}html body table td,html body table th{border:1px solid #d6d6d6;padding:6px 13px}html body dl{padding:0}html body dl dt{padding:0;margin-top:16px;font-size:1em;font-style:italic;font-weight:700}html body dl dd{padding:0 16px;margin-bottom:16px}html body code{font-family:Menlo,Monaco,Consolas,'Courier New',monospace;font-size:.85em;color:#000;background-color:#f0f0f0;border-radius:3px;padding:.2em 0}html body code::after,html body code::before{letter-spacing:-.2em;content:'\00a0'}html body pre>code{padding:0;margin:0;word-break:normal;white-space:pre;background:0 0;border:0}html body .highlight{margin-bottom:16px}html body .highlight pre,html body pre{padding:1em;overflow:auto;line-height:1.45;border:#d6d6d6;border-radius:3px}html body .highlight pre{margin-bottom:0;word-break:normal}html body pre code,html body pre tt{display:inline;max-width:initial;padding:0;margin:0;overflow:initial;line-height:inherit;word-wrap:normal;background-color:transparent;border:0}html body pre code:after,html body pre code:before,html body pre tt:after,html body pre tt:before{content:normal}html body blockquote,html body dl,html body ol,html body p,html body pre,html body ul{margin-top:0;margin-bottom:16px}html body kbd{color:#000;border:1px solid #d6d6d6;border-bottom:2px solid #c7c7c7;padding:2px 4px;background-color:#f0f0f0;border-radius:3px}@media print{html body{background-color:#fff}html body h1,html body h2,html body h3,html body h4,html body h5,html body h6{color:#000;page-break-after:avoid}html body blockquote{color:#5c5c5c}html body pre{page-break-inside:avoid}html body table{display:table}html body img{display:block;max-width:100%;max-height:100%}html body code,html body pre{word-wrap:break-word;white-space:pre}}.markdown-preview{width:100%;height:100%;box-sizing:border-box}.markdown-preview ul{list-style:disc}.markdown-preview ul ul{list-style:circle}.markdown-preview ul ul ul{list-style:square}.markdown-preview ol{list-style:decimal}.markdown-preview ol ol,.markdown-preview ul ol{list-style-type:lower-roman}.markdown-preview ol ol ol,.markdown-preview ol ul ol,.markdown-preview ul ol ol,.markdown-preview ul ul ol{list-style-type:lower-alpha}.markdown-preview .newpage,.markdown-preview .pagebreak{page-break-before:always}.markdown-preview pre.line-numbers{position:relative;padding-left:3.8em;counter-reset:linenumber}.markdown-preview pre.line-numbers>code{position:relative}.markdown-preview pre.line-numbers .line-numbers-rows{position:absolute;pointer-events:none;top:1em;font-size:100%;left:0;width:3em;letter-spacing:-1px;border-right:1px solid #999;-webkit-user-select:none;-moz-user-select:none;-ms-user-select:none;user-select:none}.markdown-preview pre.line-numbers .line-numbers-rows>span{pointer-events:none;display:block;counter-increment:linenumber}.markdown-preview pre.line-numbers .line-numbers-rows>span:before{content:counter(linenumber);color:#999;display:block;padding-right:.8em;text-align:right}.markdown-preview .mathjax-exps .MathJax_Display{text-align:center!important}.markdown-preview:not([data-for=preview]) .code-chunk .code-chunk-btn-group{display:none}.markdown-preview:not([data-for=preview]) .code-chunk .status{display:none}.markdown-preview:not([data-for=preview]) .code-chunk .output-div{margin-bottom:16px}.markdown-preview .md-toc{padding:0}.markdown-preview .md-toc .md-toc-link-wrapper .md-toc-link{display:inline;padding:.25rem 0}.markdown-preview .md-toc .md-toc-link-wrapper .md-toc-link div,.markdown-preview .md-toc .md-toc-link-wrapper .md-toc-link p{display:inline}.markdown-preview .md-toc .md-toc-link-wrapper.highlighted .md-toc-link{font-weight:800}.scrollbar-style::-webkit-scrollbar{width:8px}.scrollbar-style::-webkit-scrollbar-track{border-radius:10px;background-color:transparent}.scrollbar-style::-webkit-scrollbar-thumb{border-radius:5px;background-color:rgba(150,150,150,.66);border:4px solid rgba(150,150,150,.66);background-clip:content-box}html body[for=html-export]:not([data-presentation-mode]){position:relative;width:100%;height:100%;top:0;left:0;margin:0;padding:0;overflow:auto}html body[for=html-export]:not([data-presentation-mode]) .markdown-preview{position:relative;top:0;min-height:100vh}@media screen and (min-width:914px){html body[for=html-export]:not([data-presentation-mode]) .markdown-preview{padding:2em calc(50% - 457px + 2em)}}@media screen and (max-width:914px){html body[for=html-export]:not([data-presentation-mode]) .markdown-preview{padding:2em}}@media screen and (max-width:450px){html body[for=html-export]:not([data-presentation-mode]) .markdown-preview{font-size:14px!important;padding:1em}}@media print{html body[for=html-export]:not([data-presentation-mode]) #sidebar-toc-btn{display:none}}html body[for=html-export]:not([data-presentation-mode]) #sidebar-toc-btn{position:fixed;bottom:8px;left:8px;font-size:28px;cursor:pointer;color:inherit;z-index:99;width:32px;text-align:center;opacity:.4}html body[for=html-export]:not([data-presentation-mode])[html-show-sidebar-toc] #sidebar-toc-btn{opacity:1}html body[for=html-export]:not([data-presentation-mode])[html-show-sidebar-toc] .md-sidebar-toc{position:fixed;top:0;left:0;width:300px;height:100%;padding:32px 0 48px 0;font-size:14px;box-shadow:0 0 4px rgba(150,150,150,.33);box-sizing:border-box;overflow:auto;background-color:inherit}html body[for=html-export]:not([data-presentation-mode])[html-show-sidebar-toc] .md-sidebar-toc::-webkit-scrollbar{width:8px}html body[for=html-export]:not([data-presentation-mode])[html-show-sidebar-toc] .md-sidebar-toc::-webkit-scrollbar-track{border-radius:10px;background-color:transparent}html body[for=html-export]:not([data-presentation-mode])[html-show-sidebar-toc] .md-sidebar-toc::-webkit-scrollbar-thumb{border-radius:5px;background-color:rgba(150,150,150,.66);border:4px solid rgba(150,150,150,.66);background-clip:content-box}html body[for=html-export]:not([data-presentation-mode])[html-show-sidebar-toc] .md-sidebar-toc a{text-decoration:none}html body[for=html-export]:not([data-presentation-mode])[html-show-sidebar-toc] .md-sidebar-toc .md-toc{padding:0 16px}html body[for=html-export]:not([data-presentation-mode])[html-show-sidebar-toc] .md-sidebar-toc .md-toc .md-toc-link-wrapper .md-toc-link{display:inline;padding:.25rem 0}html body[for=html-export]:not([data-presentation-mode])[html-show-sidebar-toc] .md-sidebar-toc .md-toc .md-toc-link-wrapper .md-toc-link div,html body[for=html-export]:not([data-presentation-mode])[html-show-sidebar-toc] .md-sidebar-toc .md-toc .md-toc-link-wrapper .md-toc-link p{display:inline}html body[for=html-export]:not([data-presentation-mode])[html-show-sidebar-toc] .md-sidebar-toc .md-toc .md-toc-link-wrapper.highlighted .md-toc-link{font-weight:800}html body[for=html-export]:not([data-presentation-mode])[html-show-sidebar-toc] .markdown-preview{left:300px;width:calc(100% - 300px);padding:2em calc(50% - 457px - 300px / 2);margin:0;box-sizing:border-box}@media screen and (max-width:1274px){html body[for=html-export]:not([data-presentation-mode])[html-show-sidebar-toc] .markdown-preview{padding:2em}}@media screen and (max-width:450px){html body[for=html-export]:not([data-presentation-mode])[html-show-sidebar-toc] .markdown-preview{width:100%}}html body[for=html-export]:not([data-presentation-mode]):not([html-show-sidebar-toc]) .markdown-preview{left:50%;transform:translateX(-50%)}html body[for=html-export]:not([data-presentation-mode]):not([html-show-sidebar-toc]) .md-sidebar-toc{display:none}
/* Please visit the URL below for more information: */
/* https://shd101wyy.github.io/markdown-preview-enhanced/#/customize-css */
</style>
<!-- The content below will be included at the end of the <head> element. --><script type="text/javascript">
document.addEventListener("DOMContentLoaded", function () {
// your code here
});
</script></head><body for="html-export">
<div class="crossnote markdown-preview ">
<p><strong>姓名</strong>:朱梓涵<br>
<strong>学号</strong>24325356</p>
<h2 id="1-实验目的"><strong>1. 实验目的</strong> </h2>
<p>本实验旨在深入理解 RISC-V 指令集架构 (ISA) 与单周期 CPU 的工作原理。实验目标包括:</p>
<ol>
<li>设计并实现一个支持部分 RV32I 指令集的单周期 CPU。</li>
<li>掌握 CPU 经典五阶段(取指、译码、执行、访存、写回)在单周期模型下的数据通路和控制逻辑设计。</li>
<li>学习使用 Chisel 硬件描述语言进行模块化硬件设计。</li>
<li>通过编写单元测试和集成测试,利用 <code>chiseltest</code> 和 GTKWave 波形图验证 CPU 各模块及整体功能的正确性。</li>
</ol>
<h2 id="2-实验环境"><strong>2. 实验环境</strong> </h2>
<ul>
<li><strong>操作系统</strong>: Windows 11</li>
<li><strong>开发工具</strong>: IntelliJ IDEA</li>
<li><strong>构建工具</strong>: SBT</li>
<li><strong>仿真与测试</strong>: Verilator, GTKWave</li>
</ul>
<h3 id="3-阶段功能划分"><strong>3. 阶段功能划分</strong> </h3>
<ol>
<li><strong>取指 (IF)</strong>: 程序计数器 (PC) 提供指令地址,指令存储器根据该地址输出 32 位指令。PC 在每个时钟周期默认加 4若遇跳转或分支成功则更新为目标地址。</li>
<li><strong>译码 (ID)</strong>: 译码单元解析指令,提取操作数(寄存器地址 rs1, rs2、目标寄存器地址 (rd) 和立即数。主控制器根据指令操作码生成后续阶段所需的所有控制信号。</li>
<li><strong>执行 (EX)</strong>: 算术逻辑单元 (ALU) 根据译码阶段生成的控制信号,对来自寄存器文件或立即数生成器的操作数进行计算。同时,分支跳转的地址计算和条件判断也在此阶段完成。</li>
<li><strong>访存 (MEM)</strong>: 根据译码阶段生成的访存控制信号与数据存储器进行交互。Load 指令从此阶段读取数据Store 指令则向此阶段写入数据。</li>
<li><strong>写回 (WB)</strong>: 将执行结果或从数据存储器中读取的数据写回寄存器文件。写回的数据来源由控制信号决定。</li>
</ol>
<h2 id="4-模块实现与分析"><strong>4. 模块实现与分析</strong> </h2>
<h3 id="41-取指"><strong>4.1 取指</strong> </h3>
<ul>
<li>
<p><strong>功能</strong>: 根据 <code>PC</code> 的当前值从指令存储器中取出指令,并计算下一周期的 <code>PC</code> 值。</p>
</li>
<li>
<p><strong>实现要点</strong>: <code>PC</code> 的更新逻辑是核心。它由跳转标志 <code>jump_flag_id</code> 控制:若跳转发生,<code>PC</code> 更新为 <code>jump_address_id</code>;否则,顺序执行,<code>PC</code> 更新为 <code>PC + 4</code></p>
</li>
<li>
<p><strong>代码实现</strong>:</p>
<pre data-role="codeBlock" data-info="scala" class="language-scala scala"><code><span class="token comment">// lab1(InstructionFetch)</span>
when<span class="token punctuation">(</span>io<span class="token punctuation">.</span>jump_flag_id<span class="token punctuation">)</span> <span class="token punctuation">{</span>
pc <span class="token operator">:</span><span class="token operator">=</span> io<span class="token punctuation">.</span>jump_address_id
<span class="token punctuation">}</span><span class="token punctuation">.</span>otherwise <span class="token punctuation">{</span>
pc <span class="token operator">:</span><span class="token operator">=</span> pc <span class="token operator">+</span> <span class="token number">4.</span>U
<span class="token punctuation">}</span>
<span class="token comment">// lab1(InstructionFetch) end</span>
</code></pre></li>
<li>
<p><strong>波形图</strong></p>
</li>
</ul>
<p><img src="b1.png" alt="b1"></p>
<h3 id="42-译码"><strong>4.2 译码</strong> </h3>
<ul>
<li>
<p><strong>功能</strong>: 解析指令,生成立即数,并为后续阶段提供控制信号。</p>
</li>
<li>
<p><strong>实现要点</strong>: 控制信号的生成逻辑是关键。例如,<code>memory_read_enable</code> 仅在 L-type 指令时有效,<code>wb_reg_write_source</code> 根据指令类型选择写回数据是来自 ALU、数据存储器还是 PC+4。</p>
</li>
<li>
<p><strong>代码实现</strong>:</p>
<pre data-role="codeBlock" data-info="scala" class="language-scala scala"><code><span class="token comment">// lab1(InstructionDecode)</span>
io<span class="token punctuation">.</span>ex_aluop2_source <span class="token operator">:</span><span class="token operator">=</span> Mux<span class="token punctuation">(</span>opcode <span class="token operator">==</span><span class="token operator">=</span> InstructionTypes<span class="token punctuation">.</span>RM<span class="token punctuation">,</span>
ALUOp2Source<span class="token punctuation">.</span>Register<span class="token punctuation">,</span>
ALUOp2Source<span class="token punctuation">.</span>Immediate
<span class="token punctuation">)</span>
io<span class="token punctuation">.</span>memory_read_enable <span class="token operator">:</span><span class="token operator">=</span> opcode <span class="token operator">==</span><span class="token operator">=</span> InstructionTypes<span class="token punctuation">.</span>L
io<span class="token punctuation">.</span>memory_write_enable <span class="token operator">:</span><span class="token operator">=</span> opcode <span class="token operator">==</span><span class="token operator">=</span> InstructionTypes<span class="token punctuation">.</span>S
io<span class="token punctuation">.</span>wb_reg_write_source <span class="token operator">:</span><span class="token operator">=</span> MuxLookup<span class="token punctuation">(</span>
opcode<span class="token punctuation">,</span>
RegWriteSource<span class="token punctuation">.</span>ALUResult
<span class="token punctuation">)</span><span class="token punctuation">(</span>
IndexedSeq<span class="token punctuation">(</span>
InstructionTypes<span class="token punctuation">.</span>L <span class="token operator">-&gt;</span> RegWriteSource<span class="token punctuation">.</span>Memory<span class="token punctuation">,</span>
Instructions<span class="token punctuation">.</span>jal <span class="token operator">-&gt;</span> RegWriteSource<span class="token punctuation">.</span>NextInstructionAddress<span class="token punctuation">,</span>
Instructions<span class="token punctuation">.</span>jalr <span class="token operator">-&gt;</span> RegWriteSource<span class="token punctuation">.</span>NextInstructionAddress
<span class="token punctuation">)</span>
<span class="token punctuation">)</span>
<span class="token comment">// lab1(InstructionDecode) end</span>
</code></pre></li>
<li>
<p><strong>波形图</strong></p>
</li>
</ul>
<p><img src="b2.png" alt="b2"></p>
<h3 id="43-执行"><strong>4.3 执行</strong> </h3>
<ul>
<li>
<p><strong>功能</strong>: 执行 ALU 运算,并处理分支和跳转逻辑。</p>
</li>
<li>
<p><strong>实现要点</strong>: ALU 的两个操作数 <code>op1</code><code>op2</code> 的来源由译码模块生成的 <code>aluop*_source</code> 信号决定,实现了操作数的灵活选择。</p>
</li>
<li>
<p><strong>代码实现</strong>:</p>
<pre data-role="codeBlock" data-info="scala" class="language-scala scala"><code><span class="token comment">// lab1(Execute)</span>
alu<span class="token punctuation">.</span>io<span class="token punctuation">.</span>func <span class="token operator">:</span><span class="token operator">=</span> alu_ctrl<span class="token punctuation">.</span>io<span class="token punctuation">.</span>alu_funct
alu<span class="token punctuation">.</span>io<span class="token punctuation">.</span>op1 <span class="token operator">:</span><span class="token operator">=</span> Mux<span class="token punctuation">(</span>
io<span class="token punctuation">.</span>aluop1_source <span class="token operator">==</span><span class="token operator">=</span> ALUOp1Source<span class="token punctuation">.</span>Register<span class="token punctuation">,</span>
io<span class="token punctuation">.</span>reg1_data<span class="token punctuation">,</span>
io<span class="token punctuation">.</span>instruction_address
<span class="token punctuation">)</span>
alu<span class="token punctuation">.</span>io<span class="token punctuation">.</span>op2 <span class="token operator">:</span><span class="token operator">=</span> Mux<span class="token punctuation">(</span>
io<span class="token punctuation">.</span>aluop2_source <span class="token operator">==</span><span class="token operator">=</span> ALUOp2Source<span class="token punctuation">.</span>Register<span class="token punctuation">,</span>
io<span class="token punctuation">.</span>reg2_data<span class="token punctuation">,</span>
io<span class="token punctuation">.</span>immediate
<span class="token punctuation">)</span>
<span class="token comment">// lab1(Execute) end</span>
</code></pre></li>
<li>
<p><strong>波形图</strong></p>
</li>
</ul>
<p><img src="b3.png" alt="b3"></p>
<h3 id="44-cpu"><strong>4.4 CPU</strong> </h3>
<ul>
<li>
<p><strong>功能</strong>: 实例化取指、译码、执行等所有子模块,并根据数据通路图将它们正确连接起来。</p>
</li>
<li>
<p><strong>实现要点</strong>: 确保数据流和控制流在模块间的正确传递是顶层连接的关键。例如,从取指模块输出的指令和指令地址,需要同时送往译码和执行模块。</p>
</li>
<li>
<p><strong>代码实现</strong>:</p>
<pre data-role="codeBlock" data-info="scala" class="language-scala scala"><code><span class="token comment">// lab1(cpu)</span>
ex<span class="token punctuation">.</span>io<span class="token punctuation">.</span>instruction <span class="token operator">:</span><span class="token operator">=</span> inst_fetch<span class="token punctuation">.</span>io<span class="token punctuation">.</span>instruction
ex<span class="token punctuation">.</span>io<span class="token punctuation">.</span>instruction_address <span class="token operator">:</span><span class="token operator">=</span> inst_fetch<span class="token punctuation">.</span>io<span class="token punctuation">.</span>instruction_address
ex<span class="token punctuation">.</span>io<span class="token punctuation">.</span>reg1_data <span class="token operator">:</span><span class="token operator">=</span> regs<span class="token punctuation">.</span>io<span class="token punctuation">.</span>read_data1
ex<span class="token punctuation">.</span>io<span class="token punctuation">.</span>reg2_data <span class="token operator">:</span><span class="token operator">=</span> regs<span class="token punctuation">.</span>io<span class="token punctuation">.</span>read_data2
ex<span class="token punctuation">.</span>io<span class="token punctuation">.</span>immediate <span class="token operator">:</span><span class="token operator">=</span> id<span class="token punctuation">.</span>io<span class="token punctuation">.</span>ex_immediate
ex<span class="token punctuation">.</span>io<span class="token punctuation">.</span>aluop1_source <span class="token operator">:</span><span class="token operator">=</span> id<span class="token punctuation">.</span>io<span class="token punctuation">.</span>ex_aluop1_source
ex<span class="token punctuation">.</span>io<span class="token punctuation">.</span>aluop2_source <span class="token operator">:</span><span class="token operator">=</span> id<span class="token punctuation">.</span>io<span class="token punctuation">.</span>ex_aluop2_source
<span class="token comment">// lab1(cpu) end</span>
</code></pre></li>
</ul>
<h2 id="5-测试与结果分析"><strong>5. 测试与结果分析</strong> </h2>
<h3 id="51-单元测试instructiondecodertest-分析"><strong>5.1 单元测试InstructionDecoderTest 分析</strong> </h3>
<ul>
<li>
<p><strong>L-Type 指令测试分析</strong>:</p>
<ul>
<li>
<p>测试用例首先向译码模块的 <code>io.instruction</code> 端口输入一条 L-Type 指令。随后,测试代码使用 <code>.expect()</code> 方法断言模块的各个输出端口信号值是否与预设的期望值一致。</p>
</li>
<li>
<p>对于 <code>lw</code> 指令,其核心功能是从内存读取数据并写回寄存器。因此:</p>
<ul>
<li><code>memory_read_enable</code> 必须为 <code>true</code>,以启动访存阶段的数据读取。</li>
<li><code>reg_write_enable</code> 必须为 <code>true</code>,以允许最终结果写入寄存器。</li>
<li><code>wb_reg_write_source</code> 必须是 <code>RegWriteSource.Memory</code>,因为写回的数据源于数据存储器。</li>
<li><code>ex_immediate</code> 必须是指令编码中包含的偏移量(此例中为 32用于地址计算 <code>rs1 + imm</code><br>
这些 <code>expect</code> 值精确地定义了 L-Type 指令在译码阶段应生成的正确行为模式,确保了控制通路的正确性。</li>
</ul>
</li>
</ul>
</li>
<li>
<p><strong>波形图分析</strong>:</p>
<p><img src="b2.png" alt="图 2 lw"></p>
<p>图片展示了 <code>lw x10, 32(x5)</code> 指令(机器码 <code>0x02028503</code>)的译码过程。在时钟上升沿之后:</p>
<ol>
<li><code>id_io_instruction</code> 端口稳定为 <code>0x02028503</code></li>
<li><code>id_io_memory_read_enable</code><code>id_io_reg_write_enable</code> 信号被置为高电平 (1)。</li>
<li><code>id_io_wb_reg_write_source</code> 信号变为 <code>01</code> (二进制),对应 <code>RegWriteSource.Memory</code></li>
<li><code>id_io_ex_immediate</code> 的值为 <code>0x00000020</code>,即十进制的 32。<br>
波形图直观地验证了译码模块对于 L-Type 指令的响应与 <code>expect</code> 语句的预期完全一致。</li>
</ol>
</li>
</ul>
<h3 id="52-整体测试cputest-以fibonaccitest-为例"><strong>5.2 整体测试CPUTest (以<code>FibonacciTest</code> 为例)</strong> </h3>
<ul>
<li>
<p><strong><code>fibonacci.c</code> 程序分析</strong>:</p>
<ul>
<li><strong>程序功能</strong>:<br>
该 C 语言程序定义了一个递归函数 <code>fib(int a)</code>,用于计算斐波那契数列的第 <code>a</code> 项。其逻辑为:如果 <code>a</code> 等于 1 或 2则返回 1否则返回 <code>fib(a-1)</code><code>fib(a-2)</code> 之和。<code>main</code> 函数是程序的入口点,它调用 <code>fib(10)</code> 来计算斐波那契数列的第 10 项,并将最终结果(即 55存储到内存地址 <code>0x4</code><pre data-role="codeBlock" data-info="c" class="language-c c"><code><span class="token keyword keyword-int">int</span> <span class="token function">fib</span><span class="token punctuation">(</span><span class="token keyword keyword-int">int</span> a<span class="token punctuation">)</span> <span class="token punctuation">{</span>
<span class="token keyword keyword-if">if</span> <span class="token punctuation">(</span>a <span class="token operator">==</span> <span class="token number">1</span> <span class="token operator">||</span> a <span class="token operator">==</span> <span class="token number">2</span><span class="token punctuation">)</span> <span class="token keyword keyword-return">return</span> <span class="token number">1</span><span class="token punctuation">;</span>
<span class="token keyword keyword-return">return</span> <span class="token function">fib</span><span class="token punctuation">(</span>a <span class="token operator">-</span> <span class="token number">1</span><span class="token punctuation">)</span> <span class="token operator">+</span> <span class="token function">fib</span><span class="token punctuation">(</span>a <span class="token operator">-</span> <span class="token number">2</span><span class="token punctuation">)</span><span class="token punctuation">;</span>
<span class="token punctuation">}</span>
<span class="token keyword keyword-int">int</span> <span class="token function">main</span><span class="token punctuation">(</span><span class="token punctuation">)</span> <span class="token punctuation">{</span>
<span class="token operator">*</span><span class="token punctuation">(</span><span class="token keyword keyword-int">int</span> <span class="token operator">*</span><span class="token punctuation">)</span><span class="token punctuation">(</span><span class="token number">4</span><span class="token punctuation">)</span> <span class="token operator">=</span> <span class="token function">fib</span><span class="token punctuation">(</span><span class="token number">10</span><span class="token punctuation">)</span><span class="token punctuation">;</span> <span class="token comment">// 将 fib(10) 的结果写入内存地址 0x4</span>
<span class="token punctuation">}</span>
</code></pre></li>
<li><strong>Chisel 测试检查方式</strong>:<br>
测试框架首先将 <code>fibonacci.asmbin</code> 文件加载到 CPU 的指令存储器和数据存储器中。然后,它驱动 CPU 执行足够多的时钟周期(通过 <code>c.clock.step(1000)</code> 大循环),以确保递归计算和内存写入操作全部完成。在仿真结束时,测试代码通过读取数据存储器的调试端口:<pre data-role="codeBlock" data-info="scala" class="language-scala scala"><code>c<span class="token punctuation">.</span>io<span class="token punctuation">.</span>mem_debug_read_address<span class="token punctuation">.</span>poke<span class="token punctuation">(</span><span class="token number">4.</span>U<span class="token punctuation">)</span> <span class="token comment">// 将调试地址设置为 0x4</span>
c<span class="token punctuation">.</span>clock<span class="token punctuation">.</span>step<span class="token punctuation">(</span><span class="token punctuation">)</span> <span class="token comment">// 步进一个时钟周期</span>
c<span class="token punctuation">.</span>io<span class="token punctuation">.</span>mem_debug_read_data<span class="token punctuation">.</span>expect<span class="token punctuation">(</span><span class="token number">55.</span>U<span class="token punctuation">)</span> <span class="token comment">// 期望从地址 0x4 读出的数据为 55</span>
</code></pre>上述语句首先将调试端口的读取地址 <code>mem_debug_read_address</code> 设置为 <code>0x4</code>,然后步进一个时钟周期以便数据稳定,最后断言 <code>mem_debug_read_data</code> 的值是否等于预期的 <code>55</code>。如果相等,则测试通过,验证了 CPU 能够正确执行递归计算并将结果写入指定内存。</li>
</ul>
</li>
<li>
<p><strong>波形图分析</strong>:</p>
<p><img src="b41.png" alt="图 3 "></p>
<p>图片展示了 <code>fibonacci</code> 程序执行的最后几个周期以及结果检查阶段。</p>
<ol>
<li><strong>执行分析</strong>: 在波形图的前半部分,可以看到 CPU 正在执行 <code>main</code> 函数中对 <code>fib(10)</code> 的调用,并最终将结果存储到内存地址 <code>0x4</code>。由于斐波那契数列的递归特性PC (<code>inst_fetch_io_instruction_address</code>) 会在函数调用和返回的过程中不断跳转。在程序即将结束时PC 会最终指向退出或停机指令。</li>
<li><strong>结果检查</strong>: <code>main</code> 函数的核心是 <code>*(int *)(4) = fib(10);</code> 这条语句,它会在 <code>fib(10)</code> 的计算结果(即 55准备好后将其写入数据存储器的 <code>0x4</code> 地址。<br>
在波形图的<strong>最后几个周期</strong>,我们会观察到以下关键信号变化:
<ul>
<li><code>mem_io_memory_write_enable</code> 信号短暂置高(为 <code>1</code>),表明数据存储器发生了写入操作。</li>
<li><code>mem_io_memory_bundle_address</code> 信号会显示为 <code>0x00000004</code>,确认写入目标是内存地址 <code>0x4</code></li>
<li><code>mem_io_memory_bundle_write_data[31:0]</code> 信号会显示为 <code>0x00000037</code>(十六进制的 55这是 <code>fib(10)</code> 的计算结果。</li>
<li>在这之后,由于测试使用 <code>c.io.mem_debug_read_address.poke(4.U)</code> 来读取内存,波形图上 <code>mem_io_debug_read_address</code> 会变为 <code>0x4</code>,同时 <code>io_mem_debug_read_data</code>(或类似输出调试信号)会稳定显示出 <code>0x00000037</code><br>
这些波形特征与 Chisel 测试中 <code>mem_debug_read_data.expect(55.U)</code> 的断言完美吻合,证明 CPU 正确地执行了 <code>fibonacci</code> 程序并存储了预期结果。</li>
</ul>
</li>
</ol>
</li>
</ul>
<h2 id="6-遇到的问题与改进建议"><strong>6. 遇到的问题与改进建议</strong> </h2>
<ol>
<li>
<p><strong>问题:立即数生成规则复杂,指导文档未详细展开。</strong><br>
在实现译码模块时RISC-V 不同指令格式I/S/B/U/J的立即数拼接方式各不相同实验指导对此描述较为简略导致初期实现困难。</p>
<ul>
<li><strong>建议</strong>:在实验指导中为每种指令格式提供一个具体的立即数拼接图示或示例代码,能极大帮助理解。</li>
</ul>
</li>
<li>
<p><strong>问题:缺少对波形图调试的引导案例。</strong><br>
初次使用 GTKWave 进行硬件调试时,面对海量的信号线,难以快速定位关键信号。</p>
<ul>
<li><strong>建议</strong>:实验文档中可以附加一个简单的调试案例,例如追踪一条 <code>add</code> 指令的数据流,展示如何从取指 PC 开始,逐步添加 <code>instruction</code>, <code>reg_read_data</code>, <code>alu_result</code>, <code>reg_write_data</code> 等信号,并解释它们在波形图上的时序关系。</li>
</ul>
</li>
<li>
<p><strong>问题:部分模块接口和控制信号的设计意图不够明确。</strong><br>
例如,<code>ALUOp1Source</code><code>ALUOp2Source</code> 这类控制信号的设计初衷,需要通过阅读多个模块的代码才能完全理解其作用。</p>
<ul>
<li><strong>建议</strong>:在代码框架的注释中,对关键的 <code>object</code><code>Enum</code>(如 <code>ALUOp1Source</code>),增加注释来说明每个选项的用途和对应的指令场景。</li>
</ul>
</li>
</ol>
<h2 id="7-实验结论"><strong>7. 实验结论</strong> </h2>
<p>通过本次实验,我成功设计并实现了一个功能基本完备的 RISC-V 单周期 CPU。在实现过程中我深入理解了数据通路与控制信号在指令执行过程中的协同作用并掌握了模块化的硬件设计思想。通过编写和分析单元测试与集成测试我学会了如何利用 <code>chiseltest</code> 和波形图对硬件设计进行验证和调试。本次实验极大地加深了我对计算机组成原理中理论知识的实践理解,为后续更复杂的处理器设计打下了坚实的基础。</p>
<p><span class="katex-display"><span class="katex"><span class="katex-mathml"><math xmlns="http://www.w3.org/1998/Math/MathML" display="block"><semantics><mrow><mi>e</mi><mo>=</mo><mi>m</mi><msup><mi>c</mi><mn>2</mn></msup></mrow><annotation encoding="application/x-tex">e=mc^2</annotation></semantics></math></span><span class="katex-html" aria-hidden="true"><span class="base"><span class="strut" style="height:0.4306em;"></span><span class="mord mathnormal">e</span><span class="mspace" style="margin-right:0.2778em;"></span><span class="mrel">=</span><span class="mspace" style="margin-right:0.2778em;"></span></span><span class="base"><span class="strut" style="height:0.8641em;"></span><span class="mord mathnormal">m</span><span class="mord"><span class="mord mathnormal">c</span><span class="msupsub"><span class="vlist-t"><span class="vlist-r"><span class="vlist" style="height:0.8641em;"><span style="top:-3.113em;margin-right:0.05em;"><span class="pstrut" style="height:2.7em;"></span><span class="sizing reset-size6 size3 mtight"><span class="mord mtight">2</span></span></span></span></span></span></span></span></span></span></span></span></p>
</div>
</body></html>