当前位置:网站首页>吃透Chisel语言.29.Chisel进阶之通信状态机(一)——通信状态机:以闪光灯为例
吃透Chisel语言.29.Chisel进阶之通信状态机(一)——通信状态机:以闪光灯为例
2022-07-30 13:47:00 【github-3rr0r】
Chisel进阶之通信状态机(一)——通信状态机:以闪光灯为例
上一部分我们学习了单个有限状态机的Chisel描述方法,但是单个有限状态机通常很难描述稍微有点复杂的数字设计。在这种情况下,可以把问题划分为两个或更多的小的、简单的FSM。这个FSM之间使用信号进行通信,某个FSM的输出是另一个FSM的输入,这个FSM又观察另一个FSM的输出。把一个大的FSM分割成简单的小FSM的做法叫作分解FSM。通信FSM通常直接根据设计规格直接设计,因为如果用首先单个FSM实现设计的话就太大了。这一篇文章就一起来学习通信状态机的设计实现。
通信FSM——以闪光灯为例
为了讨论通信FSM,我们使用闪光灯的例子,这个闪光灯有一个输入start和一个输出light,该闪光灯的设计规格如下:
- 当
start在一个周期内为高电平时,闪烁开始; - 会闪烁三次;
- 一次闪烁中,
light为on持续6个周期,为off持续4个周期; - 序列之后,FSM会将
light置为off并等待下一个start;
直接用一个FSM实现这个闪光灯的话,该FSM会有27中状态:1个是等待输入的初始状态, 3 × 6 3\times 6 3×6个关于on状态的状态, 2 × 4 2\times 4 2×4个关于off状态的状态,共27中状态。我们这里就不写这种实现的代码了。
那我们可以把这个大FSM分解为两个小FSM:一个主FSM用于实现闪烁逻辑,一个定时器FSM用于实现等待,下图展示了这两个FSM之间的组合:

定时器FSM会倒数6个周期或4个周期,用于生成想要的时序,定时器的规格如下:
- 当
timerLoad被设置的时候,定时器加载一个值到倒数计数器,不依赖于状态; timerSelect会选择在加载5或3;timerDone会在计数器完成倒数的时候被设置并保持被设置状态;- 其他情况,定时器会进行倒数。
下面的代码就展示了该闪光灯的定时器FSM和主FSM实现:
val timerReg = RegInit(0.U)
// 计时器连接
val timerLoad = WireDefault(false.B) // 给load信号后启动定时器
val timerSelect = WireDefault(true.B) // 选择6周期循环或4周期循环
val timerDone = Wire(Bool())
timerDone := timerReg === 0.U
timerLoad := timerDone
// 定时器FSM(倒数计数器)
when(!timerDone) {
timerReg := timerReg - 1.U
}
when(timerLoad) {
when(timerSelect) {
timerReg := 5.U
} .otherwise {
timerReg := 3.U
}
}
val off :: flash1 :: space1 :: flash2 :: space2 :: flash3 :: Nil = Enum(6)
val stateReg = RegInit(off)
val light = WireDefault(false.B) // FSM的输出
// 主FSM
switch(stateReg) {
is(off) {
timerLoad := true.B
timerSelect := true.B
when (start) {
stateReg := flash1}
}
is (flash1) {
timerSelect := false.B
light := true.B
when (timerDone) {
stateReg := space1}
}
is (space1) {
when (timerDone) {
stateReg := flash2}
}
is (flash2) {
timerSelect := false.B
light := true.B
when (timerDone) {
stateReg := space2}
}
is (space2) {
when (timerDone) {
stateReg := flash3}
}
is (flash3) {
timerSelect := false.B
light := true.B
when (timerDone) {
stateReg := off}
}
}
代码就不解释了,就提示一点,switch语句里面没有赋值的wire都会赋默认值,所有space1和space2里面可以省略timerSelect和light的赋值,但建议最好别这么写。
不过这个解决方案在主FSM中还是有冗余代码,flash1、flash2和flash3状态的功能一致,space1和space2也是如此。我们可以把这几种flash状态分解到第二个计数器里面,那么主FSM就只有三个状态了off、flash和space。下图就展示了新的通信FSM设计:

现在图里面有两个计数FSM了,一个用于计数on和off间隔的时钟周期数,另一个用于计数剩下的flash次数。下面的代码就是向下计数FSM部分:
val cntReg = RegInit(0.U)
cntDone := cntReg === 0.U
// 向下计数FSM
when(cntLoad) {
cntReg := 2.U}
when(cntDecr) {
cntReg := cntReg - 1.U}
注意计数器加载了2以闪烁三次,因为它会倒数剩下的闪烁次数,并在定时器完成是递减到space状态。下面的代码展示了完整的新的闪光灯FSM实现:
import chisel3._
import chisel3.util._
class SimpleFsm extends Module {
val io = IO(new Bundle {
val start = Input(Bool())
val light = Output(Bool())
val state = Output(UInt()) // 用于调试
})
val light = WireDefault(false.B)
val timerLoad = WireDefault(false.B)
val timerSelect = WireDefault(true.B)
val timerDone = Wire(Bool())
val timerReg = RegInit(0.U)
timerDone := timerReg === 0.U
timerLoad := timerDone
when(!timerDone) {
timerReg := timerReg - 1.U
}
when(timerLoad) {
when(timerSelect) {
timerReg := 5.U
}.otherwise {
timerReg := 3.U
}
}
val cntLoad = WireDefault(false.B)
val cntDecr = WireDefault(true.B)
val cntDone = Wire(Bool())
val cntReg = RegInit(0.U)
cntDone := cntReg === 0.U
when(cntLoad) {
cntReg := 2.U }
when(cntDecr) {
cntReg := cntReg - 1.U }
val off :: flash :: space :: Nil = Enum(3)
val stateReg = RegInit(off)
switch(stateReg) {
is(off) {
timerLoad := true.B
timerSelect := true.B
cntLoad := true.B
when(io.start) {
stateReg := flash }
}
is(flash) {
cntLoad := false.B
timerSelect := false.B
light := true.B
when(timerDone & !cntDone) {
stateReg := space }
when(timerDone & cntDone) {
stateReg := off }
}
is(space) {
cntDecr := timerDone
when(timerDone) {
stateReg := flash }
}
}
io.light := light
io.state := stateReg
}
object MyModule extends App {
println(getVerilogString(new SimpleFsm()))
}
除了将主FSM的状态减少为3中以外,我们新的配置也更加可配置了。没有任何一个FSM会需要在我们想改变on、off时长的时候被修改。
测试代码如下:
import chisel3._
import chiseltest._
import org.scalatest.flatspec.AnyFlatSpec
class SimpleTestExpect extends AnyFlatSpec with ChiselScalatestTester {
"DUT" should "pass" in {
test(new SimpleFsm) {
dut =>
dut.clock.step()
println(dut.io.state.peekInt(), dut.io.light.peekBoolean())
dut.io.start.poke(true.B)
for (a <- 0 until 50) {
dut.clock.step()
dut.io.start.poke(false.B)
println(a, dut.io.state.peekInt(), dut.io.light.peekBoolean())
}
}
}
}
观察测试的输出,可知测试通过。
结语
这一篇文章我们以闪光灯电路为例,探索了通信电路,尤其是通信FSM的写法,FSM之间仅交换了控制信号。不过电路之间还可以交换数据,对于数据的协调交换,我们需要使用握手信号,这一部分的最后一篇文章我们就会一起学习用于单向数据交换的控制流的ready-valid接口。下一篇文章,我们将会学习带数据通路的状态机,从手动指定ready-valid信号开始。
边栏推荐
- A new generation of open source free terminal tools, so cool
- kubernate部署服务
- 产品年度营销计划书
- Learning notes - 7 weeks as data analyst "in the first week: data analysis of thinking"
- 简单理解精确率(Precision),召回率(Recall),准确率(Accuracy),TP,TN,FP,FN
- Logic Vulnerability----Permission Vulnerability
- [ARC092D] Two Faced Edges
- eclipse连接SQL server数据库「建议收藏」
- Eight years of testing experience, why was the leader criticized: the test documents you wrote are not as good as those of fresh graduates
- 机器学习在竞赛和工业界应用区别
猜你喜欢
随机推荐
20220729 Securities, Finance
mongodb打破原则引入SQL,它到底想要干啥?
权威推荐!腾讯安全DDoS边缘安全产品获国际研究机构Omdia认可
CF1677E Tokitsukaze and Beautiful Subsegments
【VMware虚拟机安装mysql5.7教程】
LeetCode二叉树系列——145.二叉树的后序遍历
电池包托盘有进水风险,存在安全隐患,紧急召回52928辆唐DM
canvas彩虹桥动画js特效
机器学习在竞赛和工业界应用区别
jsArray array copy method performance test 2207300040
Study Notes - Becoming a Data Analyst in Seven Weeks "Week 2: Business": Business Analysis Metrics
简单理解精确率(Precision),召回率(Recall),准确率(Accuracy),TP,TN,FP,FN
手把手教你写让人眼前一亮的软件测试简历,收不到面试邀请算我输
[VMware virtual machine installation mysql5.7 tutorial]
CF603E Pastoral Oddities
创意loadingjs特效小点跳跃动画
js人均寿命和GDP散点图统计样式
#第九章 子查询课后习题
八年测试经验,为何惨遭领导痛批:你写的测试文档还不如刚来的应届生
什么是缺陷分析?一篇文章带你了解,测试工程师必备技能









