Mips 处理器仿真设计
采用 Verilog 语言开发
Window 10 环境下
使用了 Sublime Text + ModelSim 的方式进行开发(ModelSim 自带的编辑器真的难用一个tab竟然是8个空格啊啊啊啊啊)
Project 1 已完成支持9条指令的单周期。 Project 2 目前已完成支持9条指令的流水,暂不支持转发、阻塞和异常。
单周期的 MIPS 处理器
只支持 add、sub、and、or、lw、sw、slt、beq、j 这9条指令
具体分为两大部分和处理器部分
定义了各种元部件
模块名:alu
说明:算逻部件
输入接口:op(4位,运算符编码), a, b(32位,运算数)
输出接口:zero(结果是否为0), dout(32位,运算结果)
op的说明:
0010: dout = a + b
0110: dout = a - b
0001: dout = a | b
0000: dout = a & b
0111: dout = a < b ? 1 : 0
模块名:dm_4k
说明:数据寄存器,大小为4k(时钟上升沿触发)
输入接口:addr(10位,数据地址), din(32位,写数据时的数据端), we(写数据使能端), re(读数据使能端), clk(时钟端)
输出接口:dout(32位,读数据时的数据输出端)
模块名:ext
说明:符号扩展部件(W 表示输入数据宽度)
输入接口:din(W 位)
输出接口:dout(32位)
模块名:im_4k
说明:指令存储器,大小为4k
输入接口:addr(10位,运算符编码)
输出接口:dout(32位,对应指令)
模块名:mux2
说明:二路选择器(W 表示输入数据宽度)
输入接口:a, b(W 位,表示0和1对应的数据源), s(选择信号)
输出接口:dout(W 位,选择结果)
模块名:pc
说明:程序计数器(时钟上升沿触发)
输入接口:clk(时钟端), rst(重置信号), data(32位,下一指令地址)
输出接口:dout(32位,当前指令地址)
模块名:regheap
说明:寄存器堆(时钟上升沿触发)
输入接口:clk(时钟端), we(写寄存器使能端), rreg1(5位,读寄存器1地址), rreg2(5位,读寄存器2地址), wreg(5位,写寄存器的地址), wdata(写入寄存器的数据)
输出接口:rdata1(32位,读寄存器1的数据), rdata2(32位,读寄存器2的数据)
另:有部分为方便测试而添加的初始化寄存器的值的代码。
解析指令,生成对应的控制信号
模块名:ALUctrl
说明:算逻部件控制器
输入接口:ALUOp(2位), funct(6位,指令的5-0位)
输出接口:op(4位,对应的 alu 运算符编码)
模块名:ctrl
说明:算逻部件控制器
输入接口:op(6位,指令的31-26位)
输出接口:RegDst, RegWrite, ALUSrc, MemRead, MemWrite, MemtoReg, Jump, Branch, ALUOp(各种控制信号,其中 ALUOp 为2位)
控制信号名 | 无效时的含义 | 有效时的含义 |
---|---|---|
RegDst | 写寄存器的目标寄存器号来自 rt 字段(位 20 : 16 ) | 写寄存器的目标寄存器号来自 rd 字段(位 15 : 11 ) |
RegWrite | 无 | 寄存器堆写使能有效 |
ALUSrc | 第二个 ALU 操作数来自寄存器堆的第二个输出(读数据2) | 第二个 ALU 操作数为指令低16位的符号扩展 |
MemRead | 无 | 数据存储器读使能有效 |
MemWrite | 无 | 数据存储器写使能有效 |
MemroReg | 写入存储器的数据来自 ALU | 写入寄存器的数据来自数据存储器 |
Jump | 无 | PC 跳转至指定地址 |
Branch | 无 | PC 在分支条件成立时跳转 |
模块名:mips
说明:单周期处理器(时钟上升沿触发)
输入接口:clk(时钟端), rst(重置信号)
模块名:testbench
说明:生成时钟信号测试部件可行性
流水级 MIPS 处理器
暂时支持 add、sub、and、or、lw、sw、slt、beq、j、addi 这10条指令
支持转发、阻塞,暂不支持异常
和 Project_1 相比,差别在于添加了中间寄存器和转发、阻塞机制
模块名:IF_ID
说明:IF/ID(时钟上升沿触发)
输入接口:clk(时钟端), en(修改使能端), rst(重置信号), IF_pc_plus_4(32位,pc + 4), IF_ins(32位,IF取出的指令)
输出接口:ID_pc_plus_4(32位), ID_ins(32位)
模块名:ID_EX
说明:ID/EX(时钟上升沿触发)
输入接口:clk(时钟端),
ID_RegDst, ID_RegWrite, ID_MemRead, ID_MemWrite, ID_ALUSrc, ID_MemtoReg, ID_ALUOp(除 ID_ALUOp 为2位外其余均为1位,表示对应的控制信号),
ID_rdata1(32位,rs 寄存器的数据), ID_rdata2(32位,rt 寄存器的数据), ID_const_or_addr(32位,I 指令中的地址/立即数),
ID_rs(5位, 表示 rs 寄存器), ID_rt(5位, 表示 rt 寄存器), ID_rd(5位, 表示 rd 寄存器)
输出接口:EX_RegDst, EX_RegWrite, EX_MemRead, EX_MemWrite, EX_ALUSrc, EX_MemtoReg, EX_ALUOp,
EX_rdata1, EX_rdata2, EX_const_or_addr,
EX_rs, EX_rt, EX_rd(分别表示对应的数据位)
模块名:EX_MEM
说明:EX/MEM(时钟上升沿触发)
输入接口:clk(时钟端),
EX_MemtoReg , EX_RegWrite , EX_MemRead , EX_MemWrite(表示对应的控制信号),
EX_ALU_res(32位,ALU 运算结果), EX_rdata2(32位,rt 寄存器的数据), EX_wreg(5位,写入的寄存器地址)
输出接口:MEM_MemtoReg, MEM_RegWrite, MEM_MemRead, MEM_MemWrite,
MEM_ALU_res, MEM_rdata2, MEM_wreg(分别表示对应的数据位)
模块名:MEM_WB
说明:MEM/WB(时钟上升沿触发)
输入接口:clk(时钟端),
MEM_MemtoReg, MEM_RegWrite(表示对应的控制信号),
MEM_rdata(32位,从 dm 中读取的数据), MEM_ALU_res(ALU 运算结果), MEM_wreg(5位,写入的寄存器地址)
输出接口:WB_MemtoReg, WB_RegWrite,
WB_rdata, WB_ALU_res, WB_wreg(分别表示对应的数据位)
*模块名:ctrl
*说明:新增对 addi 指令的支持
模块名:fu
说明:转发单元
输入接口:EX_rs, EX_rt, MEM_RegWrite, MEM_rd, WB_RegWrite, WB_rd
输出接口:ForwardA, ForwardB
多选器控制 | 源 | 解释 |
---|---|---|
ForwardA = 00 | ID/EX | 第一个 ALU 操作数来自寄存器堆 |
ForwardA = 10 | EX/MEM | 第一个 ALU 操作数由上一个 ALU 运算结果转发获得 |
ForwardA = 01 | MEM/WB | 第一个 ALU 操作数从数据存储器或者前面的 ALU 结果中转发获得 |
ForwardB = 00 | ID/EX | 第二个 ALU 操作数来自寄存器堆 |
ForwardB = 10 | EX/MEM | 第二个 ALU 操作数由上一个 ALU 运算结果转发获得 |
ForwardB = 01 | MEM/WB | 第二个 ALU 操作数从数据存储器或者前面的 ALU 结果中转发获得 |
模块名:hdu
说明:load-use 数据冒险检测单元
输入接口:EX_MemRead, EX_rt, ID_rs, ID_rt
输出接口:stall(阻塞信号)
模块名:br_hdu
说明:分支冒险检测单元
输入接口:Branch, EX_RegWrite, EX_wreg, MEM_MemRead, MEM_wreg, ID_rs, ID_rt
输出接口:stall(阻塞信号)
模块名:mux3
说明:三路选择器(W 表示输入数据宽度)
输入接口:a, b, c(W 位,表示0和1对应的数据源), s(2位,选择信号)
输出接口:dout(W 位,选择结果)
*模块名:pc
*说明:新增写入使能端
*新增输入接口:en(写入使能端)
*模块名:regheap
*说明:移除初始数据,全部初始化为0
注:星号表示有新增或修改