最新串口RS232通信程序Verilog.docx
- 文档编号:1141488
- 上传时间:2022-10-17
- 格式:DOCX
- 页数:13
- 大小:176.05KB
最新串口RS232通信程序Verilog.docx
《最新串口RS232通信程序Verilog.docx》由会员分享,可在线阅读,更多相关《最新串口RS232通信程序Verilog.docx(13页珍藏版)》请在冰豆网上搜索。
最新串口RS232通信程序Verilog
串口RS232通信程序(Verilog)
串口RS232通信程序(Verilog)
串口有9个管脚,其中只有三个是最重要的,分别是
pin2:
RxD(receivedata).接收数据
pin3:
TxD(transmitdata).发送数据
pin5:
GND(ground). 地
串行通信时序
我们先来看看字节0x55的发送
0x55的二进制代码是01010101,但发送时由低字节开始的,因此发送次序依次为1-0-1-0-1-0-1-0.
串行通信电平
· "1"issentusing-10V(orbetween-5Vand-15V).
· "0"issentusing+10V(orbetween5Vand15V).
由于计算机RS232的电平与电路板(通常+5V)之间电平的不同所以要用到转换芯片
如果PCB板电源+-5V的话用MAX232
如果PCB板(FPGA)电源是+-3.3V的话用MAX3232
这个图的串口如果采用母头的话,要用交叉公母线,保证是PCB板上这边的RxD连计算机的TxD(3Pin),PCB板这边的TxD连计算机的RxD(2Pin).
串行通信波特率
这里要弄清楚波特率与比特率的差别:
比特率是数字信号的传输速率,它用单位时间内传输的二进制代码的有效位(bit)数来表示,其单位为每秒比特数bit/s(bps)、每秒千比特数(Kbps)或每秒兆比特数(Mbps)来表示(此处K和M分别为1000和1000000,而不是涉及计算机存储器容量时的1024和1048576)。
波特率指数据信号对载波的调制速率,它用单位时间内载波调制状态改变次数来表示,其单位为波特(Baud)。
波特率与比特率的关系为:
比特率=波特率X单个调制状态对应的二进制位数
两相调制(单个调制状态对应1个二进制位)的比特率等于波特率;
四相调制(单个调制状态对应2个二进制位)的比特率为波特率的两倍;
八相调制(单个调制状态对应3个二进制位)的比特率为波特率的三倍;
依次类推。
对于串行通信来说,或者说是对于普通的数字电路来说,都是两相调制(单个调制状态对应1个二进制位),因此波特率=比特率(通常叫波特率)。
PS:
可以看看下面图就知道什么是四相调制。
如果系统时钟是1.8432MHz,那16分频就得到115200Hz
reg[3:
0]BaudDivCnt;
always@(posedgeclk)BaudDivCnt<=BaudDivCnt+1;
wireBaudTick=(BaudDivCnt==15);
但通常系统的时钟不是刚刚好是波特率的整数倍,如果不采用DCM对系统进行倍频的话,可以采用下面程序进行处理,设系统时钟为2MHz=2000000Hz
2000000/115200=17.36111
1024/59=17.356
2000000/115200≈1024/59
两个频率很接近,可以采用下面程序产生我们要的波特率。
//10bitsfortheaccumulator([9:
0]),andoneextrabitfortheaccumulatorcarry-out([10])
reg[10:
0]acc; //11bitstotal!
always@(posedgeclk)
acc<=acc[9:
0]+59;
//useonly10bitsfromthepreviousresult,butsavethefull11bits
wireBaudTick=acc[10];//sothatthe11thbitisthecarry-out
当系统时钟为2MHz的时候,计算得到的波特率的值为115234,与115200只有0.03%的误差。
我们怎么得到“59”呢,可以看下面的推导
其中Baud< 参照上面的程序与公式推导可以把程序修改如下: parameterClkFrequency=25000000;//25MHz parameterBaud=115200; parameterBaudGeneratorAccWidth=16; parameterBaudGeneratorInc=(Baud< reg[BaudGeneratorAccWidth: 0]BaudGeneratorAcc; always@(posedgeclk) BaudGeneratorAcc<=BaudGeneratorAcc[BaudGeneratorAccWidth-1: 0]+BaudGeneratorInc; wireBaudTick=BaudGeneratorAcc[BaudGeneratorAccWidth]; 当要注意的是,上面程序中BaudGeneratorInc的计算公式出错,因为在Verilog语言中中间结果只能32位,而这个公式计算的结果超过了32位。 所以要把这行改为 parameterBaudGeneratorInc=((Baud<<(BaudGeneratorAccWidth-4))+(ClkFrequency>>5))/(ClkFrequency>>4); 程序改变,得到的波特率不变。 RS232发送接收模块 RS-232发送模块 下面是我们所想要实现的: 它应该能像这样工作: ∙发送器接收8位的数据,并将其串行输出。 ("TxD_start"置位后开始传输). ∙当有数传输的时候,使"busy"信号有效,此时“TxD_start”信号被忽略. RS-232模块的参数是固定的: 8位数据,2个停止位,无奇偶校验. 数据串行化 假设我们已经有了一个115200波特的"BaudTick"信号. 我们需要产生开始位、8位数据以及停止位。 用状态机来实现看起来比较合适。 reg[3: 0]state; always@(posedgeclk) case(state) 4'b0000: if(TxD_start)state<=4'b0100; 4'b0100: if(BaudTick)state<=4'b1000;//开始位 4'b1000: if(BaudTick)state<=4'b1001;//bit0 4'b1001: if(BaudTick)state<=4'b1010;//bit1 4'b1010: if(BaudTick)state<=4'b1011;//bit2 4'b1011: if(BaudTick)state<=4'b1100;//bit3 4'b1100: if(BaudTick)state<=4'b1101;//bit4 4'b1101: if(BaudTick)state<=4'b1110;//bit5 4'b1110: if(BaudTick)state<=4'b1111;//bit6 4'b1111: if(BaudTick)state<=4'b0001;//bit7 4'b0001: if(BaudTick)state<=4'b0010;//停止位1 4'b0010: if(BaudTick)state<=4'b0000;//停止位2 default: if(BaudTick)state<=4'b0000; endcase 注意看这个状态机是怎样实现当"TxD_start"有效就开始,但只在"BaudTick"有效的时候才转换状态的。 . 现在,我们只需要产生"TxD"输出即可. regmuxbit; always@(state[2: 0]) case(state[2: 0]) 0: muxbit<=TxD_data[0]; 1: muxbit<=TxD_data[1]; 2: muxbit<=TxD_data[2]; 3: muxbit<=TxD_data[3]; 4: muxbit<=TxD_data[4]; 5: muxbit<=TxD_data[5]; 6: muxbit<=TxD_data[6]; 7: muxbit<=TxD_data[7]; endcase //将开始位、数据以及停止位结合起来 assignT D=(state<4)|(state[3]&muxbit); 完整的代码在可以得到。 RS232接收模块 下面是我们想要实现的模块: 我们的设计目的是这样的: 1.当RxD线上有数据时,接收模块负责识别RxD线上的数据 2.当收到一个字节的数据时,锁存接收到的数据到"data"总线,并使"data_ready"有效一个周期。 注意: 只有当"data_ready"有效时,"data"总线的数据才有效,其他的时间里不要使用"data"总线上的数据,因为新的数据可能已经改变了其中的部分数据。 过采样 异步接收机必须通过一定的机制与接收到的输入信号同步(接收端没有办法得到发送断的时钟)。 这里采用如下办法。 1.为了确定新数据的到来,即检测开始位,我们使用几倍于波特率的采样时钟对接收到的信号进行采样。 2.一旦检测到"开始位",再将采样时钟频率降为已知的发送端的波特率。 典型的过采样时钟频率为接收到的信号的波特率的16倍,这里我们使用8倍的采样时钟。 当波特率为115200时,采样时钟为921600Hz。 假设我们已经有了一个8倍于波特率的时钟信号"Baud8Tick",其频率为921600Hz。 具体设计 首先,接受到的"RxD"信号与我们的时钟没有任何关系,所以采用两个D触发器对其进行过采样,并且使之我我们的时钟同步。 reg[1: 0]RxD_sync; always@(posedgeclk)if(Baud8Tick)RxD_sync<={RxD_sync[0],RxD}; 首先我们对接收到的数据进行滤波,这样可以防止毛刺信号被误认为是开始信号。 reg[1: 0]RxD_cnt; regRxD_bit; always@(posedgeclk) if(Baud8Tick) begin if(RxD_sync[1]&&RxD_cnt! =2'b11)RxD_cnt<=RxD_cnt+1; else if(~RxD_sync[1]&&RxD_cnt! =2'b00)RxD_cnt<=RxD_cnt-1; if(RxD_cnt==2'b00)RxD_bit<=0; else if(RxD_cnt==2'b11)RxD_bit<=
- 配套讲稿:
如PPT文件的首页显示word图标,表示该PPT已包含配套word讲稿。双击word图标可打开word文档。
- 特殊限制:
部分文档作品中含有的国旗、国徽等图片,仅作为作品整体效果示例展示,禁止商用。设计者仅对作品中独创性部分享有著作权。
- 关 键 词:
- 最新 串口 RS232 通信 程序 Verilog