1.rvseed_defines.v(定义了一些参数,没有实际意义)

  该文件定义了一些基本参数,在后续的代码中都会调用该文件

// simulation clock period
`define SIM_PERIOD 20 // 20ns -> 50MHz // processor
`define CPU_WIDTH 32 // 32位的CPU // instruction memory
`define INST_MEM_ADDR_DEPTH 1024
`define INST_MEM_ADDR_WIDTH 10 // 2^10 = 1024

  因为设计的是32位的CPU,所以CPU的深度CPU_WIDTH为32。

2.inst_mem.v(指令存储器)

`include "rvseed_defines.v"
module inst_mem (
input [`CPU_WIDTH-1:0] curr_pc, // current pc addr,当前的pc值
output reg [`CPU_WIDTH-1:0] inst // instruction,输出的指令
); reg [`CPU_WIDTH-1:0] inst_mem_f [0:`INST_MEM_ADDR_DEPTH-1]; //存储器的位宽和处理器的位宽一致,深度设置为1024,因此地址位宽应该为10 //根据当前的pc值取出存储器对应地址的数据
always @(*) begin
inst = inst_mem_f[curr_pc[`INST_MEM_ADDR_WIDTH+2-1:2]];
end
endmodule

  模块的输入为curr_pc,即当前的pc值,输出为inst,是输出的指令,两个量都是32位的;

  定义了一个1024*32的一个存储器inst_mem,因为这里设计一个处理器,所以直接定义即可,这样会消耗逻辑资源,实际情况是使用现存的存储资源;

  inst从inst_mem中取指令,关于这行代码,我的理解是:

/*
这里为什么是INST_MEM_ADDR_WIDTH+2-1这样写,是因为比如一开始curr_pc的值是32个0,取他的[11:2]位,
这样就是相当于inst = inst_mem_f[0],正好取到第一行的32位的指令(inst_mem_f是一个1024*32的列表),
因此,如果是要取到第二个指令,那么就应该让curr_pc+4,也就是变成0000....010,这样他的[11:2]就成了000..01
正好取到了inst_mem_f[1],也就是下一条指令,即第二个指令
*/ /*
+2是因为指令输出只需要11:2位,后两位是数据存储所用到的(这里up主没有说,我是看评论区的)
*/

3.pc_reg.v(程序计数器)

//程序计数器,告诉处理器现在执行哪条指令
`include "rvseed_defines.v" module pc_reg (
input clk, // system clock
input rst_n, // active low reset
output reg ena, // system enable
input [`CPU_WIDTH-1:0] next_pc, // next pc addr
output reg [`CPU_WIDTH-1:0] curr_pc // current pc addr
); //控制处理器的运行和暂停
always @ (posedge clk or negedge rst_n) begin
if(~rst_n)
ena <= 1'b0;
else
ena <= 1'b1;
end always @ (posedge clk or negedge rst_n) begin
if(~rst_n)
curr_pc <= `CPU_WIDTH'b0; //复位,则回到第一条指令的位置
else
curr_pc <= next_pc; //将当前的pc值更新为外部提供的next_pc,next_pc由mux模块提供
end endmodule

  ena是使能信号,决定处理器的运行和暂停(这在下一个mux_pc模块中体现)

  第二个模块是pc值的更新,复位则回到第一条指令,否则将当前的pc值更新为next_pc。

4.mux_pc.v(多路选择器)

`include "rvseed_defines.v"

module mux_pc (
input ena,
input branch, // branch type
input zero, // alu result is zero
input jump, // jump type
input [`CPU_WIDTH-1:0] imm, // immediate
input [`CPU_WIDTH-1:0] curr_pc, // current pc addr
output reg [`CPU_WIDTH-1:0] next_pc // next pc addr
); always @(*) begin
if (~ena) //复位,使能为0无效,则pc值不做更新,相当于是在暂停
next_pc = curr_pc;
else if (branch && ~zero) // bne:分支跳转
next_pc = curr_pc + imm; //将当前的指令值和指令中的立即数相加,作为一个新的指令值的
else if (jump) // jal :跳转和链接
next_pc = curr_pc + imm; //将当前的指令值和指令中的立即数相加,作为一个新的指令值的
else //标准情况:pc值不做跳转也不做保留,则将当前的pc值加4,作为下一个pc,相当于读取下一条指令,为什么加4看inst-mem中有解释的
next_pc = curr_pc + `CPU_WIDTH'h4;
end
endmodule

  因为下一条指令next_pc会受到不同因素的影响,因此由一个多路选择器进行裁决。

Rong晔大佬教程学习(2):取指的更多相关文章

  1. .NetCore微服务Surging新手傻瓜式 入门教程 学习日志---先让程序跑起来(一)

    原文:.NetCore微服务Surging新手傻瓜式 入门教程 学习日志---先让程序跑起来(一) 写下此文章只为了记录Surging微服务学习过程,并且分享给广大想学习surging的基友,方便广大 ...

  2. .Net程序员之Python基础教程学习----列表和元组 [First Day]

    一. 通用序列操作: 其实对于列表,元组 都属于序列化数据,可以通过下表来访问的.下面就来看看序列的基本操作吧. 1.1 索引: 序列中的所有元素的下标是从0开始递增的. 如果索引的长度的是N,那么所 ...

  3. objective-c基础教程——学习小结

    objective-c基础教程——学习小结   提纲: 简介 与C语言相比要注意的地方 objective-c高级特性 开发工具介绍(cocoa 工具包的功能,框架,源文件组织:XCode使用介绍) ...

  4. JSON 教程学习进度备忘

    书签:跳过:另外跳过的内容有待跟进 __________________ 学习资源:W3School. _________________ 跳过的内容: 1. ______________ 知识点:1 ...

  5. JavaScript 教程学习进度备忘(二)

    备忘:之前,只将“JS 教程”学习完毕,这篇记录:“JS HTML DOM ”.“JS 对象”.“JS Window”.“JS 库” 书签:跳过:另外跳过的内容有待跟进 _______________ ...

  6. jfinal框架教程-学习笔记

    jfinal框架教程-学习笔记 JFinal  是基于 Java  语言的极速  WEB  + ORM  开发框架,其核心设计目标是开发迅速.代码量少.学习简单.功能强大.轻量级.易扩展.Restfu ...

  7. Webpack新手入门教程(学习笔记)

    p.p1 { margin: 0.0px 0.0px 0.0px 0.0px; text-align: center; font: 30.0px Helvetica; color: #000000 } ...

  8. 【前端,干货】react and redux教程学习实践(二)。

    前言 这篇博文接 [前端]react and redux教程学习实践,浅显易懂的实践学习方法. ,上一篇简略的做了一个redux的初级demo,今天深入的学习了一些新的.有用的,可以在生产项目中使用的 ...

  9. MapServer Tutorial——MapServer7.2.1教程学习——第一节用例实践:Example1.7 Adding a wms layer

    MapServer Tutorial——MapServer7.2.1教程学习——第一节用例实践:Example1.7 Adding a wms layer 前言 Add OGC WMS Layers( ...

  10. MapServer Tutorial——MapServer7.2.1教程学习——第一节用例实践:Example1.6 Defining Projections and Extents

    MapServer Tutorial——MapServer7.2.1教程学习——第一节用例实践:Example1.6 Defining Projections and Extents 一.前言 当在m ...

随机推荐

  1. Spring Cloud OpenFeign 的使用及踩坑指南

    目录 Feign 和OpenFeign Feign OpenFeign openFeign的优势 OpenFeign应用 1. 导入依赖 2. 使用 3. 日志配置 4. 数据压缩 OpenFeign ...

  2. SpringBoot + 自定义注解,实现用户操作日志(支持SpEL表达式)

    背景 一个成熟的系统,都会针对一些关键的操作,去创建用户操作日志. 比如: XX人创建了一条订单,订单号:XXXXXXXXX 因为操作人或者订单号是动态的,所以有些开发人员,不知道获取,就将这种操作日 ...

  3. Markdown初识

    1.标题 一级标题 ctrl+1......六级标题 ctrl+6 2.字体 加粗 ctrl+B 斜体 ctrl+I 下划线 ctrl+ U 3.引用 大于号加任意键 4.分割线 "---& ...

  4. Web端上传数据到OSS

    阿里云文档:参考文献 更正第三点:用户带着从服务器获取的数据签名和文件上传到OSS,这样做可以保证安全性.减轻服务器负担. 1.操作步骤 ①新建Bucket ②创建后更改跨域设置 这一步是保证跨域请胯 ...

  5. 熟练掌握并充分利用CSS3的新特性,更新完毕。

    1.1  尝试新颖的CSS3特性 首先,我们来看一个具体的案例.  https://code.juejin.cn/pen/7277536985772720139   1.2  CSS3新特性简介和浏览 ...

  6. 全局重写Element UI中的Message消息提示显示时长

    需求:Message消息提示显示时长过长 环境:"vue": "2.6.12"."element-ui": "^2.15.6&qu ...

  7. Java四种引用 强引用,软引用,弱引用,虚引用(转)

    强引用 : 只要引用存在,垃圾回收器永远不会回收 Object obj= new Object(); Object 对象对后面 new Object的一个强引用, 只有当obj这个被释放之后,对象才会 ...

  8. replace批量替换、表删除数据查询用法

    一. select * from baec_file where bacti='1'order by baec01; select baec02,REPLACE(baec02,'白班','A班') f ...

  9. CalledFromWrongThreadException

    更新UI的位置不正确,线程解析数据    handler. mssage 中更新 android.view.ViewRootImpl$CalledFromWrongThreadException: O ...

  10. 软件开发人员 Kubernetes 入门指南|Part 1

    Kubernetes 是一个用于部署和管理容器的编排系统.使用 Kubernetes,用户可以通过自动执行管理任务(例如在跨节点间扩展容器并在容器停止时重新启动任务),在不同环境中可靠地运行容器. K ...