CPU设计实战-协处理器访问指令的实现

目录

一 协处理器的作用与功能

1.计数寄存器和比较寄存器

2.Status寄存器

3.Cause寄存器(标号为13)

4.EPC寄存器(标号为14)

5.PRId寄存器(标号为15)

6.Config 寄存器(标号为16)-配置寄存器

二 协处理器的实现

 三 协处理器访问指令说明

四 具体实现

1.译码阶段

2.执行阶段


一 协处理器的作用与功能

我们之前实现的指令操作很多都是算术操作,CPU还需要进行一些指令集扩展的操作,比如系统控制与浮点单元计算等。MIPS架构定义了四种协处理器如下:

我们在这里只实现CP0的主要功能中的中断异常处理,它还可以进行工作状态的配置如大小端模式的切换,高速缓存控制,存储管理单元控制等。

协处理器中为每种功能定义了相应的32个32位的寄存器,主要是通过读写协处理器中的寄存器去实现相关功能。

我们的目标是实现中断异常管理以及最基本的工作状态的配置,只需要关注上面标黑的寄存器即可。

1.计数寄存器和比较寄存器

定时中断的实现是通过Count计数器寄存器和Compare寄存器来实现的。

Count寄存器是一个不停计数的32位寄存器,计数频率一般与CPU时钟频率相同,当计数达到32位无符号数的上限时,会从0开始重新计数。Count 寄存器可读、可写。
Compare寄存器是一个32位的寄存器,与Count寄存器一起完成定时中断功能。 当Count寄存器中的计数值与Compare寄存器中的值一样时, 会产生定时中断。这个中断会一直保持,直到有数据被写入Compare寄存器。Compare 寄存器可读、可写。 

2.Status寄存器

 Status寄存器也是一个32位、可读、可写的寄存器,用来控制处理器的操作模式、中断使能以及诊断状态。这里我们主要进行中断使能和基本工作状态的配置。

CU3-CU0表示协处理器是否可用(Coprocessor Usability), 分别控制协处理器CP3、CP2、 CPI、
CP0。为0时,表示相应的协处理器不可用;为1时,表示相应的协处理器可用。对于OpenMIPS
处理器而言,只有协处理器CPO,所以可以设置本字段为4'b0001。

IM7-IM0表示是否屏蔽相应中断(Interrupt Mask), 0表示屏蔽,1 表示不屏蔽,MIPS处理器可以
8个中断源,对应IM字段的8位,其中6个中断源是处理器外部硬件中断,另外2个是软件中断

中断是否能够被处理器响应是由Status寄存器与Cause寄存器共同决定的,如果Status寄存器的IM字段(中断源选择)与Cause寄存器的IP字段(外部中断挂是否发生)的相应位都为1,而且Status寄存器的IE字段(全局使能中断)也为1时,处理器才响应相应中断。
EXL表示是否处于异常级(Exception Level), 当异常发生时,会设置本字段为1,表示处理器处于异常级,此时,处理器会进入内核模式下工作,并且禁止中断
IE表示是否使能中断(Interrupt Enable),这是全局中断使能标志位。为1表示中断使能,为0表示中断禁止。

3.Cause寄存器(标号为13)


Cause寄存器主要记录最近一次异常发生的原因,也控制软件中断请求。Cause寄存器的各字段如表10-6所示,除了IP[1:0]、 IV和WP,其余字段都是只读的

4.EPC寄存器(标号为14)


EPC是异常程序计数器(Exception Program Counter),用来存储异常返回地址,一般情况下,存储发生异常的指令的地址,但是,如果发生异常的指令位于延迟槽中,那么EPC存储的是前一条转移指令的地址。因为之前我们定义,延迟槽中的指令就是转移指令之后的指令并且一定执行。所以异常真正发生在转移阶段。

5.PRId寄存器(标号为15)


PRId寄存器是处理器标志( Processor Identifier)寄存器,包含的信息有:制造商信息、处理器类型以及处理器的版本等。各个字段如表10-9所示。其中R是保留字段。

6.Config 寄存器(标号为16)-配置寄存器


Config寄存器包含了与处理器有关的各种配置和功能信息,其各个字段如下所示,大部分字段由硬件在重启时进行初始化,或定为常量。

BE
其值为1表示处理器工作在大端模式(MSB),为0表示处理器工作在小端模式(LSB)。OpenMIPS处理器工作在大端模式,所以设置本字段为1。
AT
指令集架构类型(Architecture Type)字段,当其值为2'b00时,表示MIPS32架构

现在我们可以整理一下中断的实现操作:

中断分为外部中断和定时中断

定时中断的实现是通过Count计数器寄存器和Compare寄存器来实现的。

Count寄存器是一个不停计数的32位寄存器,计数频率一般与CPU时钟频率相同,当计数达到32位无符号数的上限时,会从0开始重新计数。Count 寄存器可读、可写。


Compare寄存器是一个32位的寄存器,与Count寄存器一起完成定时中断功能。 当Count寄存器中的计数值与Compare寄存器中的值一样时, 会产生定时中断。这个中断会一直保持,直到有数据被写入Compare寄存器。Compare 寄存器可读、可写。 

外部中断是否能够被处理器响应是由Status寄存器与Cause寄存器共同决定的,如果Status寄存器的IM字段(中断源选择)与Cause寄存器的IP字段(外部中断挂是否发生)的相应位都为1,而且Status寄存器的IE字段(全局使能中断)也为1时,处理器才响应相应中断。

二 协处理器的实现

之前说过对协处理器指令实现操作本质就是去读写协处理器,所以接口类似于一个单端口的存储器,包括读写使能,读写地址,输出寄存器和外部中断的输入。 

先对寄存器做初始化,主要是对工作状态寄存器进行基本赋值:

读写操作按照读写地址进行赋值,寄存器地址按照之前的表按地址号进行定义。

写操作:

对计数器和比较寄存器进行读写之前先实现他们的基本操作定义输出信号,计数器需要一直计数,cause保留外部中断请求,其值与外部中断输入一致,定时中断请求的输出:

需要注意当写入比较寄存器时,定时中断结束。以及有些寄存器并不是可写的

读操作:

 三 协处理器访问指令说明

 这两个指令一个写一个读:

这2条指令的格式与之前已实现的指令都不同,主要特点是:指令码都为6'b010000,MIPS32 指令集架构定义为COP0类,需要借助于第21~ 25bit的值才能确定具体是哪一条指令。 此外,指令的第3~10bit为0,第0~2bit是sel域,这个域的作用取决于具体的MIPS32架构处理器,对OpenMIPS处理器而言,sel 域没有作用,不用考虑。下面分别说明mtc0、mfc0 两条指令的用法、作用。
●当指令码是6'b010000,且第21~25bit的值为5'"b00100 时,是mtc0指令。
指令用法为:mtc0 rt, rd。
指令作用为: CPR[0, rd] <- GPR[rt],将地址为rt的通用寄存器的值赋给协处理器CP0中地址为rd的寄存器。
●当指令码是 6'b010000,且第21~ 25bit的值为500000时,是mfc0指令。
指令用法为: mfc0 rt, rd。
指令作用为: GPR[rt] <- CPR[0, rd],读出协处理器CPO中地址为rd的寄存器的值,并赋给地址为rt的通用寄存器。
 

四 具体实现

 

指令的读写在执行阶段进行定义,读操作需要提前去读CP0寄存器的值并传递回来,而写操作需要把使能以及写入值与地址传递到CP0.此外新引入的寄存器同样会发生数据相关问题,还是使用数据前推进行解决。

1.译码阶段

mtc0修改CP0寄存器的值,修改的值在rt里,修改的地址在rd中,所以不需要写通用寄存器,但是需要读rt中的值。

mfc0是读出协处理器CPO中地址为rd的寄存器的值,并赋给地址为rt的通用寄存器。需要写通用寄存器,两个地址都在指令中直接用,不需要读寄存器。

2.执行阶段

要实现mfc0需要两步,先去读CP0的值,可能不是最新的值所以判断一下数据相关(如果访存和回写阶段需要写回CP0并且写回地址和我们要读的地址一样),如果是把写入的值直接传过来,记为moveres。

然后再写入:

如果是mtc0 指令,那么给出对CP0中寄存器的写信息:设置写操作信号cp0_ reg_ we. o为WriteEnable、 写入地址为指令中第11~ 15bit的值、写入的值就是译码阶段传递过来的regl_i的值,参考译码阶段可知,该值正是地址为rt的通用寄存器的值。

后面把需要写入寄存器的值一直传递到CP0寄存器就行了。
 

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/739796.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

lqb省赛日志[4/37]

一只小蒟蒻备考蓝桥杯的日志 文章目录 笔记遍历 (DFS主题) 刷题心得小结 笔记 遍历 (DFS主题) 参考 BFS 的使用场景&#xff1a;层序遍历、最短路径问题 DFS -> 层次遍历 -> 无权图的最短路径 (Dijkstra 算法平替) 实现: 用队列存储, 出队, 孩子进队 隐式图遍历: 华…

Linux纯命令行查看文本文件

处理超大文本文件时&#xff0c;你可能希望避免一次性加载整个文件&#xff0c;这可能会耗尽内存资源。以下是一些在命令行中查看大文本文件的方法&#xff0c;它们适用于Linux和Unix系统&#xff0c;包括Mac OS X&#xff0c;而在Windows中&#xff0c;你可以使用类似的工具或…

初阶数据结构之---堆的应用(堆排序和topk问题)

引言 上篇博客讲到了堆是什么&#xff0c;以及堆的基本创建和实现&#xff0c;这次我们再来对堆这个数据结构更进一步的深入&#xff0c;将讲到的内容包括&#xff1a;向下调整建堆&#xff0c;建堆的复杂度计算&#xff0c;堆排序和topk问题。话不多说&#xff0c;开启我们今…

新智元 | Stable Diffusion 3技术报告流出,Sora构架再立大功!生图圈开源暴打Midjourney和DALL·E 3?

本文来源公众号“新智元”&#xff0c;仅用于学术分享&#xff0c;侵权删&#xff0c;干货满满。 原文链接&#xff1a;Stable Diffusion 3技术报告流出&#xff0c;Sora构架再立大功&#xff01;生图圈开源暴打Midjourney和DALLE 3&#xff1f; 【新智元导读】Stability AI放…

1.Python是什么?——跟老吕学Python编程

1.Python是什么&#xff1f;——跟老吕学Python编程 Python是一种什么样的语言&#xff1f;Python的优点Python的缺点 Python发展历史Python的起源Python版本发展史 Python的价值学Python可以做什么职业&#xff1f;Python可以做什么应用&#xff1f; Python是一种什么样的语言…

网络触手获取天气数据存入mysql 项目

首先这个案例不一定能直接拿来用&#xff0c;虽然我觉得可以但是里面肯定有一些我没考虑到的地方。 有问题评论或者私信我&#xff1a; 这个案例适合我这种学生小白 获取天气数据网址&#xff1a; https://lishi.tianqi.com/xianyang/202201.html 网络触手获取天气数据代码直…

c++两种去重方法(erase+unique 和 unique)

一&#xff0c;eraseunique 适用于容器如 vector。 1.代码&#xff1a; vector<int> v;//使用unique前必须排序&#xff08;他只能删除相邻相同的元素&#xff0c;背过即可&#xff09;sort(v.begin(),v.end());// unique(v.begin(),v.end())返回的是不重复元素的下一个…

分布式事务模式:AT、TCC、Saga、XA模式

AT模式 2PC使用二阶段提交协议&#xff1a;Prepare提交事务请求&#xff0c; 我认为就是执行分布式的方法&#xff0c;当所有方法都执行完毕&#xff0c;且没有错误&#xff0c;也就是ack为yes。然后开始第二阶段&#xff1a; commit:提交事务 TCC模式和消息队列模式&#x…

[软件工具]yolo实例分割数据集转labelme的json格式

软件界面&#xff1a; YOLO实例分割数据集转LabelMe JSON格式软件是一款功能强大的数据转换工具&#xff0c;旨在将YOLO&#xff08;You Only Look Once&#xff09;实例分割数据集转换为LabelMe的JSON格式&#xff0c;以满足不同图像标注软件之间的数据共享需求。 该软件具有…

前端面试-浏览器相关

文章目录 1 浏览器安全1.1 什么是 XSS 攻击&#xff1f;1. 概念2. 攻击类型 1.2 如何防御 XSS 攻击&#xff1f;1.3 什么是 CSRF 攻击&#xff1f;1. 概念2. 攻击类型 1.4 如何防御 CSRF 攻击&#xff1f;1.5 什么是中间人攻击&#xff1f;如何防范中间人攻击&#xff1f;1.6 哪…

图论(二)之最短路问题

最短路 Dijkstra求最短路 文章目录 最短路Dijkstra求最短路栗题思想题目代码代码如下bellman-ford算法分析只能用bellman-ford来解决的题型题目完整代码 spfa求最短路spfa 算法思路明确一下松弛的概念。spfa算法文字说明&#xff1a;spfa 图解&#xff1a; 题目完整代码总结ti…

基于SpringBoot的“医院信管系统”的设计与实现(源码+数据库+文档+PPT)

基于SpringBoot的“医院信管系统”的设计与实现&#xff08;源码数据库文档PPT) 开发语言&#xff1a;Java 数据库&#xff1a;MySQL 技术&#xff1a;SpringBoot 工具&#xff1a;IDEA/Ecilpse、Navicat、Maven 系统展示 功能结构图 系统首页界面图 用户注册界面图 医生…

BUG:RuntimeError: input.size(-1) must be equal to input_size. Expected 1, got 3

出现的bug为:RuntimeError: input.size(-1) must be equal to input_size. Expected 1, got 3 出现问题的截图: 问题产生原因:题主使用pytorch调用的nn.LSTM里面的input_size和外面的数据维度大小不对。问题代码如下: self.lstm nn.LSTM(input_size, hidden_size, num_laye…

PyTorch学习笔记(三)

2.4 获取tensor中元素的数量 在PyTorch中&#xff0c;如果你有一个tensor&#xff08;张量&#xff09;&#xff0c;你可以使用numel()函数来获取tensor中所有元素的数量。numel()会返回tensor中所有元素的数量&#xff0c;不考虑tensor的维度。 下面是一个例子&#xff1a; …

面向对象设计之依赖反转原则

设计模式专栏&#xff1a;http://t.csdnimg.cn/4Mt4u 目录 1.引言 2.控制反转(loC) 3.依赖注入(DI) 4.依赖注入框架(DlFramework) 5.依赖反转原则(DIP) 6.总结 1.引言 前面讲到&#xff0c;单一职责原则和开闭原则的原理比较简单&#xff0c;但在实践中用好比较难&#x…

干货!不懂Python的math模块和random模块操作还不赶紧来学!

1.导入math模块 import math 2.向上取整&#xff1a;math.ceil() num 9.12print(math.ceil(num)) # 10 3.向下取整&#xff1a;math.floor() num1 9.99print(math.floor(num1)) # 9 4.开平方&#xff1a;math.sqrt()​​​​​​​ num2 16print(math.sqrt(num…

算法打卡day8|字符串篇02|Leetcode 28. 找出字符串中第一个匹配项的下标、459. 重复的子字符串

算法题 Leetcode 28. 找出字符串中第一个匹配项的下标 题目链接:28. 找出字符串中第一个匹配项的下标 大佬视频讲解&#xff1a;KMP理论篇 KMP代码篇 个人思路 当看到在一个串中查找是否出现过另一个串&#xff0c;那肯定是用kmp算法了; kmp比较难理解,详细理论和代码可以…

c语言大小字母转换程序

#include <stdio.h> #include <ctype.h> // 引入ctype.h库以使用toupper和tolower函数 int main() { char str[100]; int choice; printf("Enter a string: "); fgets(str, sizeof(str), stdin); // 使用fgets读取字符串&#xff0c;包括空格 pr…

【Linux】入门篇---xshell安装以及远程连接Linux(看这篇就行啦!)

&#x1f490; &#x1f338; &#x1f337; &#x1f340; &#x1f339; &#x1f33b; &#x1f33a; &#x1f341; &#x1f343; &#x1f342; &#x1f33f; &#x1f344;&#x1f35d; &#x1f35b; &#x1f364; &#x1f4c3;个人主页 &#xff1a;阿然成长日记 …

Java的单例模式

谦逊不值得可耻&#xff0c;放荡才是。 Humility is not worthy of shame, debauchery is. Java单例模式 单例模式是一种常见的设计模式&#xff0c;用于确保某个类只有一个实例&#xff0c;并提供一个全局访问点。 懒汉模式 vs 饿汉模式 懒汉模式 懒汉模式&#xff1a;在需…