由于平时也没太注重基础,这次针对我本人,自己准备了一些面试官可能问到的,以及网上别人被面试到的问题,问题的难度都不大,但你不复习也许就不行。
我的文章禁止转载!!!你可以收藏!
QQ:1183699227
资料推荐:
https://blog.csdn.net/ciscomonkey/article/details/102941101
在使用Verilog描述状态机时,通常用参数定义语句 parameter指定状态编码。常用的状态编码有三种分别是:递增二进制编码,格雷码和one-hot编码。
递增的二进制码在转台跳转条件比较复杂的时候会导致很大的组合逻辑。
格雷码,这种编码方式能够避免进入错误的状态,常用于高可靠性设计。
独热码,这种编码所占用的D触发器资源比递增二进制编码多一些,速度非常快。
个人理解:格雷码非常可靠,建议格雷码
异步串行是指UART,通用异步接收/发送。
UART包含了TTL电平的串口和RS232电平的串口。
RS232
RS232采用不平衡传输方式,即所谓的单端通讯,由于发送电平与接收电平的差仅仅为2V到3V左右,所以其共模抑制能力差,再加上双绞线上的分布电容,其传送距离最大为约15米,在115200波特率的情况下,最高速率为11.5KByte/s
RS422
RS422定义了四线接口,实际上还有一根地线。RS422支持最多10个节点,即一个主设备,其余从设备,从设备之间不能通信,所以RS422支持点对多的双向通信。RS422的最大传输距离为1219米,最大传输速率为10Mb/s
。其平衡双绞线的长度与传输速率成反比,在 100kb/s速率以下,才可能达到最大传输距离。只有在很短的距离下才能获得最高速率传输。一般100米长的双绞线上所能获得的最大传输速率仅为 1Mb/s。
RS-485
RS 485与RS232类似,只是有二线和四线,所以可以最大带有32个从属设备。
USB设备 2.0
最新的USB2.0可以达到480Mbps
,最长5米。
USB 3.0
5.0Gbps
(500MB/s),从5米突破到100米。
其实这个问题以前我就写过,这里仅仅拿出例子来,回顾一下。
https://blog.csdn.net/ciscomonkey/article/details/87104636
正数的反码就是原码,负数的反码等于原码除符号位以外所有的位取反
补码:符号位保持不变,其余位取反加1
原码:-8,-8的源码是1【1000】,取反码:1【0111】,补码:1【1000】(十进制转二进制)
原码:-4,-4的原码是1100,取反码:1011, 补码:1100(十进制转二进制)
补码:1000,-8
补码:1100,说明此数是负数,减1,得1011,取反:1100,十进制4:因此为-4.
在不超限溢出的前提下,如果计算结果为正数,那么补码与原码相同,得到的就是正确结果; 如果计算结果为负数,还需要将补码再转换成原码。
-4+2=1100+0010=1110 =1101(减1)=1010(取反)= -2
-6+7=1010+0111=10001 == = 》舍去溢出得0001
通过以上练习,足以了。只需要牢记正数的反码就是源码,负数的补码等于源码除了符号位以外所有的位取反再加1.【除符取反加1】
通过计算后的补码要转换成源码。
基本运算无非就是上述7种,上述有几处公式有误,但是图片做的好看。以国际图片为准。
下面我对其总结:
与、或、非
与非:AB条件都具备、Y不发生
或非:任意条件具备、Y不发生
异或:条件AB不同,则Y发生
同或:条件AB相同,则Y发生
那么在verilog种要实现与非、或非、异或、同或运算是不是也只需要一个符号呢?答案是不行的,我们需要用与或非三种运算符号组合起来。
为什么要化简?
当表达式复杂时,其对应的数字电路所使用的元器件就会比较复杂,减少元器件可以提高电路的可靠性。
这里就介绍卡诺图化简法吧,公式我记不住。
首先找出最小项,最小项就是乘积项在输入随机的情况下,结果为1的可能性最小。n个变量可以构成2^n个最小项。
如何写出卡诺图?
我们可以观察到最小项都是用格雷码编码的,然后序列号就是格雷码的十进制
注意,遵循A为最高位的原则
卡诺图的化简:谁变了,干掉谁的原则,其实,我们只用观察谁不变,保留谁即可
掌握了上述如何化简卡诺图后,接下来,我来说明一下根据表达式,如何画出卡诺图吧
其实就是将只要有表达式出现的那一项,都写上1。
对于一般的表达式,需要将其转换为最小项的形式。
补充:公式法如何转换为最小项。
其中反演律最为重要,利用反演律可以化简为最小项。然后利用卡诺图化简。
上面给了个练习题,具体卡诺图化简步骤,这里不再赘述了,我做过,和上述结果一样。
在数字电路中,根据逻辑功能的不同,我们可以将数字电路分为2大类,一类叫做组合咯及电路,另一类叫时序逻辑电路。
输出只与当前的输入有关系,而与输入信号的上一个状态没有关系。
而时序电路是有记忆功能的,它在任何时刻的输出,不仅与该时刻的输入信号有关,而且还与该时刻以前的输入信号有关。
若输入变量在卡诺图的圈内发生改变时候,则输出不会有冒险现象,若输入变量是在相邻卡诺圈的相邻处发生变化,并且从一个卡诺圈进入到另一个卡诺圈,则可能产生逻辑冒险。
通过增加冗余项来消除逻辑冒险。
有两个以上同时变化比如0001变到0111,可能走0101这条路,或者走0011这条路。所以两者是有差别的,如果走0011这条路就会发生功能冒险,解决方法只有增加选通信号或者输出电容。(个人建议是寄存2拍)
信号由于经由不同路径传输达到某一汇合点的时间有先有后的现象,就称之为竞争,由于竞争现象所引起的电路输出发生瞬间错误的现象,就称之为冒险。FPGA设计中最简单的避免方法就是尽量使用时序逻辑同步输入输出。
加滤波电容,消除毛刺的影响;
加选通信号,避开毛刺
增加冗余项,消除逻辑冒险
常用的存储电路有两类,一类是锁存器,另一类是触发器,它们两者所采用的电路结构形式不同,信号的触发方式也不同,其中,采用电平触发方式的叫做锁存器,而采用脉冲边沿触发方式的叫做触发器。
锁存器(latch)是数字电路中的一种具有记忆功能的逻辑元件。锁存器对脉冲电平敏感的存储单元电路,它只在输入脉冲的高(低)电平期间对输入信号敏感并改变状态。
这里介绍一种D锁存器,多说两句,当控制门等于1时,此时锁存器的状态是由激励输入端D来确定的,并且D等于什么,锁存器的状态就是什么。
触发器(Flip-Flop)是对脉冲边沿敏感的存储单元电路,它只在触发脉冲的上升沿(或者下降沿)瞬间改变其状态。
非阻塞逻辑用<=
使用阻塞逻辑可以在几个ns内完成
这个问题,我们首先要知道异或门,就是相异为1,相同为0.
用异或门实现非门:只需将输入与高电平进行异或就行了。
module FPGA_mianshi
(
input a,
input b,
output q
);
assign q=a?1:b;
endmodule
用序列检测机,检测一串数字序列:
1101
//检测序列1101
module FPGA_mianshi (
input clk,
input rst,
input data, //输入串行数据流
output reg valid //代表检测到输入数据
);
localparam s1=3'b001;
localparam s2=3'b011;
localparam s3=3'b010;
localparam s4=3'b110;
reg [2:0] now_state=s1;
reg [2:0] next_state=s1;
always @ (posedge clk or negedge rst)
begin
if(!rst)
now_state<=s1;
else
now_state<=next_state;
end
always @ (*) //实现条件转换
begin
case(now_state)
s1:
if(data==1)
begin
next_state=s2;
end
else
begin
next_state=s1;
end
s2:
if(data==1)
begin
next_state=s3;
end
else
next_state=s1;
s3:
if(data==0)
begin
next_state=s4;
end
else
begin
next_state=s3;
end
s4:
if(data==1)
begin
next_state=s1;
end
else
next_state=s1;
default:next_state=s1;
endcase
end
always @ (posedge clk or negedge rst)
begin
if(!rst)
valid<=0;
else
begin
if((now_state==s4)&(data==1))
valid<=1;
else
valid<=0;
end
end
endmodule
这种题目一定要先考虑出状态图,比如,我这个1101,每种状态无非要么0,要么1.
比如已经出了 11了,那么接下来检测第三个数据是1还是0,如果是0,那么继续,如果是1,我们就要回到初始状态吗?答案是否定的,因为111,能够证明已经检测到了11两个连续了,所以,此时我们要停留在此状态,如果还是1,那么就成了1111,也满足了检测到两个1,直到我们检测到0,才迈向下一个状态。
所以序列检测,千万不要认为简单,这些细节,稍不注意,你可能出现漏检的情况。
同步逻辑是时钟之间有固定的因果关系,异步逻辑是各时钟时间没有固定的因果关系。
同步逻辑必须要收到时钟脉冲的控制;异步逻辑,只要输入变化,输出就立即变化。
电路设计的难点在于时序设计,时序设计的实质就是满足每一个触发器的建立/保持时间而要求的
建立时间:触发器在时钟上升沿到来之前,其数据输入端的数据必须保持不变的时间。
保持时间:触发器在时钟上升沿到来之后,其数据输入端的数据必须保持不变的时间。
因为触发器内部数据的形成是需要一定的时间的,如果不满足建立和保持时间,触发器将进入亚稳态,进入亚稳态后触发器的输出将不稳定,在0和1之间变化,这时候需要经过一个恢复时间,其输出才能稳定,但稳定后的值并不一定是你的输入值。这就是为什么要用两级触发器来同步异步输入信号(这一点在检测外部输入触发,尤为有经验,这也是我之间检测外部触发信号上升沿,为什么都要用2级检测,否则不稳定,但是在系统内部,我只需要一级检测即可)。这样做可以防止由于异步输入细心好对于本级时钟可能不满足建立保持时间而使本级触发器产生的亚稳态传播到后面逻辑中,导致亚稳态的传播。
亚稳态是指触发器无法再某个规定的时间段内到达一个可以确认的状态。使用两级触发器来使异步电路同步化的电路其实叫做“一步同位器”,他只能用来对他只能用来对一位异步信号进行同步。两级触发器可以防止亚稳态传播的原理:假设第一级触发器的输入不满足建立保持时间,它在第一个脉冲沿到来后输出的数据就为亚稳态,那么在下一个脉冲沿到来之前,其输出的亚稳态数据在一段恢复时间后必须稳定下来,而稳定的数据必须满足第二级触发器的建立时间,如果都满足了,在下一个脉冲沿到来时,第二级触发器将不会出现亚稳态,因为其输入端的数据满足其建立保持时间。
这个问题,以前我专门花了不少力气写过,并从官方文档入手,这里再次拿出来回顾
https://blog.csdn.net/ciscomonkey/article/details/88877296
上述公式可以说,充分总结了如何计算建立时间余量(slack)的方法。注意这是同步系统时钟信号。
只要好好领悟,非常简单。
SRAM:Static Random-Access Memory,静态随机存取存储器,主要是供电数据就会保持,但断电就会消失。也称之为Volatile memory.
SDRAM:Synchronous Dynamic Random Access Memory,同步动态随机存储器,同步写入和读出数据的DRAM。
EEOROM:Electrically Erasable Programmable Read Only Memory,电可擦除可编程只读存储器。
DDR:Double Data Synchronous Dynamic Random Accsee Memory,双倍速率同步动态随机存储器,双倍速率传输的SDRAM,在时钟的上升沿和下降沿都可以进行数据传输。我们的电脑内存条都是DDR芯片。
FLASH:flash memory,闪存,非易失性固态存储,如制成内存条或U盘。
十进制数转换成R进制:整数部分,除R取余法,除到商为0为止,小数部分,乘R取整法,乘到积为0为止。
十进制小数转二进制
取整数部分即可。
这个问题,问的组里的师兄,还是当面讲解效率高不少,然后下来简单看了一下视频就理解了。
箭头朝里,属于NMOS,所以左边属于NMOS,当G端大于S时,D、S之间相当于开关闭合,那么D与S之间可能电流D流向S,也可能S流向D。
箭头朝外,属于PMOS,S大于G时,D与S之间相当于开关闭合
通常来讲,S一般是接的电源,D端接的是输出。
记忆:
箭头往里NMOS、箭头朝外PMOS,箭头朝谁,谁更大。与G端靠近的是源极。
例子:
参考视频:https://www.dianyuan.com/class/video_2123.html
注意上述表示符号,PMOS就是有一个圆圈。NMOS是没有圆圈的。
这一点,我查阅了诸多CSDN的blog,发现抄袭居多,很多人1是没讲清楚,2是也许作者自己也没有真正理解到位。下面,作为处女座,无法看下去了。根据自己的理解,以及我的抄袭阅读。来谈谈这个FIFO的深度计算处理跨时钟域数据问题。
在数字IC设计中,我们经常遇到这种工作场景,工作在不同时钟域下的两个模块,它们之间需要进行数据传递,为了避免数据丢失,我们会使用到FIFO。当读数据的速率小于写数据的速率时候,我们就不得不将还没有被读走的数据缓存下来,那么我们需要开多大的控件缓存这些数据呢?缓存开大了,会浪费资源,开小了,会丢失数据,如何计算最小FIFO深度。
如何理解数据突发长度?为什么FIFO能解决?
首先,我得说一下数据突发长度(burst length)这个概念,如果模块A不间断的往FIFO中写数据,模块B同样不间断的从FIFO中读数据,不同的是模块A写数据的时钟频率大于B读数据的频率,那么如果系统一直在工作的话,有一部分数据还没来得及读走,新的数据又来了,也就是说那些没有被读走的数据会越累积越多,那么FIFO的深度需要无穷大,因此只有在突发数据传输过程中讨论FIFO深度才是有意义的。
什么叫突发数据长度,那就好明白了,就是发送一段数据后,然后停下来等待。这一段数据就是突发数据长度(burst length)。
这一点,很好理解,你想一下,如果饮水机的桶装水,小口朝下,大口接一个超大的水管。如果大口不断在注入水,小口同时在放水。这样,即使同时进行。但这桶水肯定会在一段时间后溢出来。那么,想一下,如果注入水和放水同时进行,但是一旦注入水刚要满的时候,我就停止继续注入水。这样就能保持水不会溢出来,水桶达到缓存水的目的。
回到正题,也就是说我们一次传递一包数据完成后再去传递下一包数据,我们把一段时间内传递的数据个数称为burst length。
在维基百科中,burst transmission是这样解释的:In telecommunication, a burst transmission or data burst is the broadcast of a relatively high-bandwidth transmission over a short period。在通信中,被解释为在较短的时间周期内,传输高带宽的数据。
下面,为了更好的理解FIFO size的计算过程,这里在不套用计算公式的前提下来逐步计算不同场景FIFO深度,当然在本文的最后会给出FIFO深度的计算公式。
在讨论之间A模块向FIFO写数据的时钟频率为fa,模块B从FIFO读数据的时钟频率为fb。
写速率大于读速率,并且没有空闲周期,一直读与写。
假设:
计算:
写120个数据需要的时间是:
1/80(us)*120=1500 ns
读一个数据需要20ns(50MHz)
在1500ns内,读取了75个数据
还剩余数据120-75个=45个数据
也就是说在1500ns内还没有被读走的数据个数=45个
因此FIFO的最小深度为45
场景2在场景1的基础上增加了一个假设,即读比写慢两拍。这种假设是真正存在的,在异步FIFO设计中,我们需要去判断FIFO的空满来保证逻辑的正确性,判断空满标志需要去比较读写指针,而读指针与写指针处在不同的时钟域中,我们需要采用格雷码和两级同步寄存器去降低亚稳态的概率,而两级同步必然会导致空满标志位的判断至少延迟2个cycle。对于空标志位来说,将写指针同步到读时钟域至少需要花费2个时钟,而在同步这段时间内有可能还会写入新的数据,因此同步后的写指针一定小于或等于(当且仅有同步时间内没有新数据写入的情况下才会等于)当前的写指针,所以此时判断不一定是真空;同理,对于满标志位来说,将读指针同步到读时钟域至少需要花费2个时钟,而在同步这段时间内有可能还会读出新的数据,因此同步后的读指针一定小于或等于当前读指针,所以此时判断并不一定是真满。
通过上述讨论可以知道场景2的FIFO最小深度应该比场景1的FIFO最小深度45略大。
fa>fb 并且都有空闲周期
那么:
由于读数据比写数据要快,因此FIFO只起到了过时钟域的作用,FIFO的最小深度为1即可。
读的速率大于写的速率,但是读和写都有空闲期
计算:
每隔1个cycle写一次,意味着2个cycle才写一个数据
每隔3个cycle读一次,意味着4个cycle才读一个数据
写一个数据需要的时间:21/50=40ns
突发传输中,写完所有数据所需要的时间=12040=4800ns
读一个数据所需要的时间是4*1/50MHz=80ns
在4800ns内能够读走的数据个数是4800/80=60
还剩余数据120-60=60个数据
因此FIFO的最小深度为60
在上面给出的场景中,我们给出的条件都是每隔几个时钟读写一次,这种周期性读写在实际中很常见。但是在工程设计中还存在着这样一种情况,只给出数据在一段时间内的读写速率,怎么读写完全随机,这种情况我们需要考虑最坏的一种情况避免数据丢失。在最坏的情形的情况中,读写的速率应该相差最大,也就是说需要找出最大的写速率和最小的读速率。
假设:
写数据时钟频率fa=80MHz
读数据时钟频率fb=50MHz
在写时钟周期内,每100个周期就有40个数据写入FIFO
在读时钟周期内,每10个周期可以有8个数据读出FIFO
这里没有给出数据的突发长度,从假设中可以得出每100个周期就有40个数据写入FIFO,这里可能就有人会说突发长度就是40个数据,其实不是这样的,因为数据是随机写入FIFO的,需要考虑最坏的情形,即写速率最大的情形,只有如下所示的情形才是写速率最高的情形,burst length为80
其实这一点也很好理解,也就是写入数据,连续写入的持续时间最长。
那么读数据应该是怎么样的呢?读数据当然是越慢,情况越糟糕
计算:
连续写入80个数据需要的时间1/80(us)80=1000ns
由于每10个周期可以有8个数据读出FIFO;
80个数据,需要B时钟的100个周期,也就是说需要1001 /50=2000ns
也就是说读这80个数据,需要2000ns,那么就是说在1000ns的时间能够读取1000 /(20ns*10/8)=40 个数据
那么还剩下40个数据,所以需要FIFO 40个
1、从上面分析来看,求FIFO的最小深度主要有以下几个要点:
不管在什么场景下,要确定FIFO的深度,关键在于计算出在突发读2、写这段时间内有多少个数据没有被读走;
由于FIFO空满标志位的判断延迟,在实际应用中需要预留一些余量
3、求FIFO深度需要考虑最坏的情形,读写的速率应该相差最大,也就是说需要找出最大的写速率和最小的读速率。
4、下面给出网上的公式,就很好理解了,这些公式不用记忆,理解了就自然会做了。
并行转换是载入并行的数据,在同步信号为有效电平时,寄存器内的数据被逐位移出。
串并转换是在时钟驱动下,将单比特的位数据流输入寄存器,并依次诸逐位移动,直到寄存器满了为止,然后直接读取并行输出。
串行数据转并行数据的verilog:
reg [7:0]sda_reg;
//wire en;
always@(posedge clk or negedge rst_n)
begin
if(!rst_n)
begin
sda_reg<=0;
end
else
begin
if(en)
begin
sda_reg<={sda_reg[6:0],sda}; //将并转串模块输出的串行数据sda转入sda_reg的每一位(从低位到高位)
end
else
begin
sda_reg<=0;
end
end
end
串并转换器,我们看到,其实就是流水线操作,如果输入为1Gbps
那么可以转换为100M*位宽10位
其实这样来理解上图,串行时钟是1ns更新一个数据,那么10ns(100M)就更新10个数,此时10个寄存器的数据都已经更新完毕,由于此时FPGA内部时钟的速度是100M。所以10个寄存器的数据都被更新,所以,实现100M采样10个bit。从而实现了串转并的思想。接口可以跑到1G以上,但是FPGA内部时钟只能够跑到400M就差不多了。通常一些高速的AD采样都是达到了1G,所以采用串转并的思想,可以实现对高速数据的处理。
并行数据转串行数据的verilog:
流水线操作是处理高速设计中的一个常用设计手段。如果某个设计的处理流程分为若干步骤,而且整个数据处理是单流向的,即没有反馈或者迭代运算,前一个步骤的输出是下一个步骤的输入,则可以考虑采用流水线设计来提高系统的工作频率。
在 WCDMA 设计中经常使用到流水线处理的方法,如 RAKE 接收机、搜索器、前导捕获等。流水线处理方式之所以频率较高,是因为复制了处理模块,它是面积换取速度思想的又一种具体体现。
前面关于FIFO的深度计算,我已经介绍得很详细了。相信已经能够熟练掌握了,那么下面来看一下异步FIFO;
FIFO是一种先进先出的数据缓冲器,特点是没有外部的读写笛子。由于没有外部的地址信号,所以只能顺序的读写,而不能跳读。FIFO的读写是根据满和空信号设计。当FIFO满的时候不可以往里面写、当FIFO空的时候不能读数据。读FIFO的时候,内部的读指针自动加1当写FIFO的时候写指针自动加1.
什么是异步FIFO,什么又是同步FIFO呢?
异步FIFO简单的来说就是读写时钟不相同,同步FIFO就是读写的时钟相同。
使用异步FIFO可以在两个不同的时钟域之间快速而方便的传输数据,起到跨时钟域处理的作用。经常用于处理跨时钟域问题。
此外对于不同宽度的数据接口也可以采用FIFO进行缓冲,如8位输入,16位输出。
如何使用异步FIFO,请参照我的另一篇blog:
https://blog.csdn.net/ciscomonkey/article/details/99702628
格雷码的概念不用多说,就是相邻两个二进制,只有1位发生了变化。
举例:
如果采集器器采到了格雷码:1010
就要将它变为自然二进制:
0 与第四位 1 进行异或结果为 1
上面结果1与第三位0异或结果为 1
上面结果1与第二位1异或结果为 0
上面结果0与第一位0异或结果为 0
因此最终结果为:1100 这就是二进制码即十进制 12
再举例:
线性反馈移位寄存器(LFSR)是内测试电路中的最基本标准模块,作为伪随机测试码产生器
一个n阶的LFSR由n个触发器和若干个异或门组成。特点是产生伪随机序列的最大长度为2^n-1;
module test(
CLR,
CLK,
EN,
DIN, //需要置进的数
DOUT);
input CLR;
input CLK;
input EN;
input [7:0] DIN;
output DOUT;
wire DOUT;
reg din;
reg [7:0] state;
always @(posedge CLK)
begin
if(CLR)
begin
state <= 8'b00000000;
end
else
begin
if(EN)
begin
state<=DIN;
end
else
begin
state[7:1] <= state[6:0];
state[0] <= din;
end
end
end
assign DOUT = state[7];
always @(state[7] or state[5]or state[4] or state[0])
begin
din = state[7] ^ state[5]^ state[4]^ state[0];
end
endmodule
这里提一点,如果有人问你有offer了吗,就实话实说,有的话就说有了,因为这是你个人能力的体现,不然别人会以为你一个offer都没有,实力不行,可以不跟他们透露具体公司和待遇,这属于你自己的隐私,他不会强问的,所以,要自信,不要隐瞒自己的offer。
(技术宅们一定要注意,你可能技术好,但是不给你展示的机会,一定要自己主动)这里强调一点。很重要:面试不要等着面试官问什么你说什么,你要掌握好节奏,你来带着面试官按照你的思路走,多说一些自己的好处,当然我不是让你不给面试官插嘴的机会,自己把握好度,说的好,就算技术不好可能一年比别人多好几万的工资,所以面试会说很重要,非常重要,特别重要。
经过2个多星期在成都一个人奔波,一忽儿,电子科大,一忽儿,川大,备受各种打击,毕竟本科也不是211/985,研究生虽然我有自信敢说高于一般学校的平台,和川大,我也敢去竞争,但跟电子科大比,我自己还是心虚的。两星期奔波,收获了2个IC公司的offer,其他乱七八糟的小公司我就不提了,但是被某家小公司拒绝的时候,彻底清醒了我的大脑,别去小公司,他们养不起我们这种,也不会耐心等待我们的成长。年轻人,最好去个大公司,薪资低一点没关系,先把自己的平台一定要往上面提一档次,这样你才能去够得着下一个档次。
以后,我可能不会太去关注FPGA这一块了,决定从新开始我的CSDN博客,从事IC相关,当然数字处理这一块,matlab仿真这一块,我不会放弃。因为这一次面试我被感觉到了,要设计一款IC,数字信号处理需要十分精通才行,这样才会有全局观念,这些参数对于IC设计时非常重要。