CPU设计实战-加载和存储指令(2)

目录

一 ll和sc指令说明

二 ll和sc指令的实现

1 llbit寄存器

2 译码阶段

3 访存阶段

4 Load相关问题

5 流水线在取指阶段暂停


本章介绍两个比较特殊的加载存储指令ll和sc,这两个指令的存在用于实现信号量机制。

信号量机制:在多线程中为了保证某个正在运行的线程不被其他线程打扰,需要遵循RMW操作序列,这是一个规范过程:读取,修改,返回,并且此过程不能有任何打扰即为一个原子操作(一定执行且不被打扰的操作为原子操作)。

原子操作的实现方式需要基于信号量机制,semaphore是一个信号量,用来表示当前是否有进程在运行,为1表示信号量使用中,为0表示信号量空闲。进行原子操作前,使用wait函数查询semaphore的值(读取),如果为1,则等待,否则,将其置为1,开始执行原子操作(修改)。操作结束后,signal 函数将semaphore置为0(返回),这样其他线程就可以执行原子操作了。

MIPS32架构采用ll和sc这两个指令实现信号量机制,具体步骤如下:

ll指令同一般的加载指令一样,从内存中加载一个字, 但是,有一点不同,ll指令还会将处理器内部的一个链接状态位LLbit置为1,表明发生了一个链接加载操作,并将链接加载的地址保存到一个特殊寄存器LLAddr中(这个寄存器在多处理器中有作用,OpenMIPS是单处理器,所以在OpenMIPS实现过程中并没有实现LLAddr寄存器)。ll指令执行完毕后,会进行一定定的操作(如:修改加载得到的数据),然后执行sc指令。

执行sc指令时,会对从II指令开始的RMW序列进行检查,判断是否受到干扰,实际就是判断LLbit是否为1,如果没有受到任何干扰,LLbit 保持为1,那么操作是原子的,sc指令会对II指令加载数据的地址进行写回操作,并设置一个通用寄存器的值为1,表示成功, 反之不进行写回操作,并设置一个通用寄存器的值为0,表示失败。

 可以看到在这里LLbit就是一种信号量,ll和sc指令搭配实现了一种RWM操作序列,ll进行读取与修改,sc指令进行返回操作。在这个过程中通过对信号量LLbit的修改实现了原子操作。

一 ll和sc指令说明

ll指令作用为:从内存中指定的加载地址处,读取一个字节,然后符号扩展至32位,保存到地址为rt的通用寄存器中。并设置LLbit状态位为1.

sc指令作用为:如果RMW序列没有受到干扰,也就是LLbit为1,那么将地址为rt的通用寄存器的值保存到内存中指定的存储地址处,同时设置地址为rt的通用寄存器的值为1,设置LLbit为0。如果RMW序列受到了干扰,也就是LLbit为0,那么不修改内存,同时设置地址为rt的通用寄存器的值为0。

二 ll和sc指令的实现

我们可以把LLbit当做一个寄存器模块,ll指令在结束后会进行写操作,sc指令会先读它,然后再进行写rt寄存器。最后还会对LLbit寄存器进行写。LLbit寄存器同样放到回写阶段。

 需要注意的是,由于对LLbit寄存器的修改是在回写阶段最后的时钟上升沿进行的,如果直接采用LLbit模块给出的LLbit寄存器的值,可能不是正确的值,因为此时处于回写阶段的上一条指令可能会需要修改LLbit寄存器,这一问题在第6章添加HI、LO寄存器时也遇到过,解决方法还是数据前推,将回写阶段指令对LLbit寄存器的操作信息(是否进行写LLbit操作以及写入的值)前推到访存阶段,访存阶段依据这些情况,确定正确的LLbit寄存器的值,所以MEM/WB模块的输出信号wb_LLbit_we、wb_LLbit_value也要送到MEM模块,就是用来解决数据相关问题的。

1 llbit寄存器

只需要对异常情况判断以及写使能,如果有异常情况信号量更新为0。至于具体的异常处理在后面才会介绍。

2 译码阶段

 ll指令加载内存地址写入寄存器,需要写使能,需要读base地址,此外还需要修改LLbit的值,也就是要对LLbit寄存器进行写操作。

sc指令从寄存器读数据写入内存,最后还要修改LLbit寄存器和原通用寄存器,要读两个寄存器,一个base一个寄存器的值。

3 访存阶段

这是加载存储指令与特殊寄存器进行读写交互的地方,为了避免数据相关操作需要先获取一下LLbit寄存器最新值,即如果回写阶段也要写LLbit那就把写入的值给LLbit的最新值:

ll指令:需要读数据存储器一个字,需要写LLbit寄存器1

sc指令:因为是信号量机制所以要先检测信号量是否异常,无异常(为1)时再进行操作:需要写寄存器值到数据存储器,需要写寄存器为1,需要写LLbit寄存器为0

4 Load相关问题

加载操作需要写入值到寄存器,如果下一条指令就需要使用此寄存器的值会不会有影响?

答案是有的,因为load加载操作在访存阶段才获取到写入寄存器的新值,而下一条指令在译码阶段就已经获取了此寄存器的值,显然这个值是落后的。如下图:

那么是否可以使用数据前推来解决数据相关问题?

答案是不可以的。原因主要是因为load是在访存阶段才获取到新值,我们之前的数据数据相关问题在执行阶段就获取到了最新的值,所以落后了一个阶段后,就算把访存的值前推倒执行阶段,而下一条指令早在前一个译码阶段就获取了。

那么该如何解决?

既然相比之前的数据相关问题落后了一个阶段,如果我们可以让第二条指令在译码阶段如果识别到当前指令与上一条指令存在load关系就暂停一次,那么就相当于如下图:

此时访存阶段获取到新值,译码阶段正好需要取值,满足数据前推的解决条件,可以使用来解决问题。

5 流水线在取指阶段暂停

输出暂停信号之前还需要把上一条的指令回传过来好进行load相关的判定(如果上一条指令是load指令且当前指令操作的寄存器就是load写入的寄存器,那么就是load相关发生)

 具体实现:

先判断上一条指令是不是加载指令:

为什么SC指令也在这里呢?因为sc指令在最后会回写寄存器,也会更改寄存器的值。

接下来判断加载指令的目的地址是否和当前指令读的寄存器一样:

如果满足load相关条件就请求流水线暂停:

 至此,加载存储指令基本结束。

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

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

相关文章

项目的搭建与配置

vue create calendar_pro 选择如下配置选项 安装 vue3 支持 vue add vue-next package.json 关闭 eslint 检测。 vue.config.js 配置跨域同源策略。 const { defineConfig } require(vue/cli-service) module.exports defineConfig({transpileDependencies: true,devServe…

MyBatis-Plus如何娴熟运用乐观锁

欢迎来到我的博客,代码的世界里,每一行都是一个故事 MyBatis-Plus如何娴熟运用乐观锁 前言乐观锁的基本概念基本概念和原理:为何乐观锁是解决并发问题的有效手段: MyBatis-Plus中乐观锁的支持1. Version 注解:2. 配置乐…

深入浅出Redis(八):Redis的集群模式

引言 Redis是一款优秀的键值对、内存非关系型数据库,单机节点下的Redis存在无法保证高可用、容量不足等问题 上篇文章介绍的哨兵主要能够保证主从架构下Redis的可用性,但是仍然存在容量不足、推举新的主节点时不能访问Redis的问题,集群可水…

开发Chrome扩展插件

1.首先开发谷歌chrome扩展插件,没有严格的项目结构目录,但是需要保证里面有一个mainfest.json文件 (必不可少的文件)。在这个文件里有三个属性必不可少:name、version、mainfest_version; // 清单文件的版本,这个必须写…

jmap-各种option参数说明

基本情况 jmap(JVM Memory Map):作用一方面是获取dump文件(堆转储快照文件,二进制文件),它还可以获取目标Java进程的内存相关信息,包括Java堆各区域的使用情况、堆中对象的统计信息…

sheng的学习笔记-AI-多分类学习:ECOC,softmax

目录:sheng的学习笔记-AI目录-CSDN博客 基本术语: 若我们欲预测的是离散值,例如“好瓜”“坏瓜”,此类学习任务称为“分类”(classification); 若欲预测的是连续值,例如西瓜成熟度0.95、0.37,…

蓝桥杯:矩形总面积(Java)

目录 问题描述输入格式输出格式代码实现 问题描述 平面上有个两个矩形R1和R2,它们各边都与坐标轴平行。设(x1, y1)和(x2 ,y2)依次是R1的左下角和右上角坐标,(x3, y3)和(x4, y4)依次是R2的左下角和右上角坐标,请你计算R1和R2的总面积是多少? …

Jmeter(GUI模式)详细教程

前些天,领导让我做接口的压力测试。What??我从未接触过这方面,什么都不知道,一脸蒙。于是我从学习jmeter开始入手。 现在记录下来jmeter的使用步骤,希望能对大家有所帮助。 一、安装Jmeter 1、电脑安装J…

1910_野火FreeRTOS教程阅读笔记_prvStartFirstTask函数

1910_野火FreeRTOS教程阅读笔记_prvStartFirstTask函数 全部学习汇总: g_FreeRTOS: FreeRTOS学习笔记 这是教程中的一个函数,通过汇编来实现的。注释部分以及结合后面的讲解部分,可能还是有一点点细节的地方让初学者疑惑。我结合我自己的理解…

SAP金江、阎韶华、雷凡将出席“第四届ISIG-RPA、低代码、流程挖掘三大峰会

3月16日,第四届「ISIG中国产业智能大会」将在上海中庚聚龙酒店拉开序幕。本届大会由苏州市金融科技协会指导,企智未来科技(RPA中国、AIGC开放社区、LowCode低码时代)主办。大会旨在聚合每一位产业成员的力量,深入探索R…

vue router 解决路由带参数跳转时出现404问题

我的页面是从一个vue页面router跳转到另一个vue页面,并且利用windows.open() 浏览器重新创建一个页签。但是不知道为什么有时候可以有时候又不行,经过反复测试与分析,最终发现是因为有一个参数的值里包含了小数点., 小数点是浏览器合法字符,不能通过encode编码转义,于是乎…

【JavaEE初阶】 关于JVM垃圾回收

文章目录 🍃前言🎋死亡对象的判断算法🚩引用计数算法🚩可达性分析算法 🌳垃圾回收算法🚩标记-清除算法🚩复制算法🚩标记-整理算法🚩分代算法🎈哪些对象会进入…