关于rs232串口的协议和接口已经在《verilog语言RS232串口接收模块设计》这篇博客中有讲述:http://blog.csdn.net/baijingdong/article/details/20460019,
本设计结合前边做的几次工程,包括ps2键盘驱动,数码管输出等完成该设计。 该模块功能可以完成对ps2键盘的数据采集,采集数据在xilinx EXCD-1开发板的
数码管输出(输出为按键值对应的16进制ASCII码值),并将结果通过串口输出到pc机,在串口调试工具窗口显示。
该模块同样包括多个部分:
module rs_clk_gen(
clk,rst,rs_clk,rs_ena
);
input clk;//系统时钟
input rst;//复位信号
input rs_ena; //串口通信允许信号
output rs_clk; //输出允许的波特率信号时钟
parameter N1=5207;//5208,9600bps
parameter N2=2603;//2604,
reg rs_clk='b0;
reg [13:0] count='d0;//计数器
always @(posedge clk or negedge rst)
begin
if(!rst)
begin
count<='d0; //复位信号到来时,count计数器清零
end
else
if(count==N1 || !rs_ena) count<='d0;//当count计满或者无串口通信使能时count都不计数
else count<=count+'b1;//当且仅当count不为0,通信使能时count计数。
end
always @(posedge clk or negedge rst)
begin
if (!rst)
rs_clk<='b0;
else
if(count==N2&&rs_ena)//当且仅当count计数到一半且通信使能时允许时钟翻转
rs_clk<='b1; //
else
rs_clk<='b0;//使得rs_clk是一个小波峰的时钟信号,在有效信号的数据位为高电平。
end
endmodule
数据发送部分代码:
`timescale 1ns / 1ps
//////////////////////////////////////////////////////////////////////////////////
// Company: bupt abs_lab
// Engineer: mr. bai
//
// Create Date: 10:11:38 03/04/2014
// Design Name:
// Module Name: rs_send
// Project Name:
// Target Devices:
// Tool versions:
// Description:
//
// Dependencies:
//
// Revision:
// Revision 0.01 - File Created
// Additional Comments:
//
//////////////////////////////////////////////////////////////////////////////////
module rs_send(
tx_start,//发送开始信号,如按键信号,产生下降沿
data_byte_in,//待发送数据
rs_clk,//发送数据时钟
rs_ena,//发送波特率使能
tx_data_out,//串行数据输出
clk,
rst
);
input clk; //输入时钟
input rst;//复位信号
input tx_start;//允许发送信号,是一个下降沿,如按键,后面需要提取该信号下降沿;
input [7:0]data_byte_in;//待发送的一个字节信号;
input rs_clk;//数据发送时钟信号,为9600的波特率信号,其为高电平时开始发送数据;
output reg rs_ena;//波特率发生器的启动信号
output reg tx_data_out='b1;//发送出的数据;
reg[3:0] count3=4'd0;//数据发送计数器
reg[7:0] tx_data;
//共有七组信号;
//==============================================
//提取tx_start信号的下降沿,
//============================================
reg tx_start0,tx_start1,tx_start2;
wire tx_start_ena= ~tx_start1 & tx_start2;//允许发送开始信号;
always @(posedge clk or negedge rst)
begin
if(!rst)
begin
tx_start0<='b1;
tx_start1<='b1;
tx_start2<='b1;
end
else
begin
tx_start0<=tx_start;
tx_start1<=tx_start0;
tx_start2<=tx_start1;
end
end
//============下降沿提取结束===============================
reg tx_int; //发送允许后,置高
always @(posedge clk or negedge rst)
begin
if(!rst)
begin
tx_int<='b0;
rs_ena<='b0;
end
else
begin
if(tx_start_ena)
begin
tx_data<=data_byte_in;//检测到发送信号时将输入信号锁存,启动发送中断,波特率产生模块使能
tx_int<='b1;
rs_ena<='b1;
end
else
if(count3>=4'd11)//发送计数器计满,停止发送中断,波特率模块停止工作
begin
tx_int<='b0;
rs_ena<='b0;
// count3<=4'd0;
end
end
end
always @(posedge clk or negedge rst)
begin
if(!rst)
tx_data_out<='b1; //串口在无有效数据输出时,一直为高电平
else
if(tx_int)
begin
if(rs_clk)//发送中断且发送时钟有效时,开始发送数据
begin
case (count3)
4'd1: tx_data_out<='b0;
4'd2: tx_data_out<=data_byte_in[0];
4'd3: tx_data_out<=data_byte_in[1];
4'd4: tx_data_out<=data_byte_in[2];//先发送一个低电平,然后发送低位,后发生高位数据
4'd5: tx_data_out<=data_byte_in[3];
4'd6: tx_data_out<=data_byte_in[4];
4'd7: tx_data_out<=data_byte_in[5];
4'd8: tx_data_out<=data_byte_in[6];
4'd9: tx_data_out<=data_byte_in[7];
4'd10: tx_data_out<=1'b1;//这里没有加奇数偶校验位;
default:tx_data_out<=1'b1;//无有效数据输出时,一直为高电平
endcase
end
// else if(count3>='d11) count3<='d0;
end
end
always @(posedge clk or negedge rst)
begin
if(!rst)
count3<='d0;
else if(!tx_int ||count3=='d11)//在非发送中断和发送完成时,发送计数器都应该清零
count3<='d0;
else if(rs_clk) count3<=count3+'b1;//发送波特率时钟有效时,每发送一个数据,计数器加1;
end
endmodule
ps2键盘驱动代码:
module ps2_driever(
clk,rst_n,ps2k_clk,ps2k_data,sm_bit,sm_seg,ps2_state,ps2_byte);
input clk; //50M时钟信号
input rst_n; //复位信号
input ps2k_clk; //PS2接口时钟信号
input ps2k_data; //PS2接口数据信号
//wire [7:0] ps2_byte; // 1byte键值,只做简单的按键扫描
output ps2_state; //键盘当前状态,ps2_state=1表示有键被按下
output reg [1:0] sm_bit='b01;
output reg [7:0]sm_seg;
output [7:0]ps2_byte;
//------------------------------------------
reg ps2k_clk_r0,ps2k_clk_r1,ps2k_clk_r2; //ps2k_clk状态寄存器
//wire pos_ps2k_clk; // ps2k_clk上升沿标志位
wire neg_ps2k_clk; // ps2k_clk下降沿标志位
//设备发送向主机的数据在下降沿有效,首先检测PS2k_clk的下降沿
//利用上面逻辑赋值语句可以提取得下降沿,neg_ps2k_clk为高电平时表示数据可以被采集
always @ (posedge clk or negedge rst_n) begin
if(!rst_n) begin
ps2k_clk_r0 <= 1'b0;
ps2k_clk_r1 <= 1'b0;
ps2k_clk_r2 <= 1'b0;
end
else begin //锁存状态,进行滤波
ps2k_clk_r0 <= ps2k_clk;
ps2k_clk_r1 <= ps2k_clk_r0;
ps2k_clk_r2 <= ps2k_clk_r1;
end
end
assign neg_ps2k_clk = ~ps2k_clk_r1 & ps2k_clk_r2; //下降沿
//-----------------数据采集-------------------------
/*
帧结构:设备发往主机数据帧为11比特,(主机发送数据包为12bit)
1bit start bit ,This is always 0,
8bit data bits,
1 parity bit,(odd parity)校验位,奇校验,
data bits 为偶数个1时该位为1,
data bits 为奇数个1时该位为0.
1bit stop bit ,this is always 1.
num 范围为 'h00,'h0A;
*/
reg[7:0] ps2_byte_r; //PC接收来自PS2的一个字节数据存储器
reg[7:0] temp_data; //当前接收数据寄存器
reg[3:0] num; //计数寄存器
always @ (posedge clk or negedge rst_n) begin
if(!rst_n) begin
num <= 4'd0;
temp_data <= 8'd0;
end
else if(neg_ps2k_clk) begin //检测到ps2k_clk的下降沿
case (num)
/*
帧结构中数据位为一个字节,且低位在前,高位在后,
这里要定义一个buf,size is one Byte.
*/
4'd0: num <= num+1'b1;
4'd1: begin
num <= num+1'b1;
temp_data[0] <= ps2k_data; //bit0
end
4'd2: begin
num <= num+1'b1;
temp_data[1] <= ps2k_data; //bit1
end
4'd3: begin
num <= num+1'b1;
temp_data[2] <= ps2k_data; //bit2
end
4'd4: begin
num <= num+1'b1;
temp_data[3] <= ps2k_data; //bit3
end
4'd5: begin
num <= num+1'b1;
temp_data[4] <= ps2k_data; //bit4
end
4'd6: begin
num <= num+1'b1;
temp_data[5] <= ps2k_data; //bit5
end
4'd7: begin
num <= num+1'b1;
temp_data[6] <= ps2k_data; //bit6
end
4'd8: begin
num <= num+1'b1;
temp_data[7] <= ps2k_data; //bit7
end
4'd9: begin
num <= num+1'b1; //奇偶校验位,不做处理
end
4'd10: begin
num <= 4'd0; // num清零
end
default: ;
endcase
end
end
reg key_f0; //松键标志位,置1表示接收到数据8'hf0,再接收到下一个数据后清零
reg ps2_state_r; //键盘当前状态,ps2_state_r=1表示有键被按下
//+++++++++++++++数据处理开始++++++++++++++++=============
always @ (posedge clk or negedge rst_n) begin //接收数据的相应处理,这里只对1byte的键值进行处理
if(!rst_n) begin
key_f0 <= 1'b0;
ps2_state_r <= 1'b0;
end
else if(num==4'd10) ///一帧数据是否采集完。
begin //刚传送完一个字节数据
if(temp_data == 8'hf0) key_f0 <= 1'b1;//判断该接收数据是否为断码
else
begin
//========================理解困难==================================
if(!key_f0)
begin //说明有键按下
ps2_state_r <= 1'b1;
ps2_byte_r <= temp_data; //锁存当前键值
end
else
begin
ps2_state_r <= 1'b0;
key_f0 <= 1'b0;
end
//=====================================================
end
end
end
/*+++++++++++++等效写法+++++++++++++++++++++++++++++
reg key_released;//收到码段后是否松开
reg [7:0] ps2_byte;
always @(posedge clk or negedge rst)
begin
if(!rst)
key_released<='b0;
else if(cnt=='h0A)//一帧数据是否采集完。
begin
if(ps2_byte_buf==8'hF0)//数据为段码f0
key_released<='b1;//松开标志位置位
else
key_released<='b0;
end
end
always @ (posedge clk or negedge rst)
begin
if(!rst)
key_pressed<= 0;
else if (cnt == 4'hA) // 采集完一个字节?
begin
if (!key_released) // 有键按过?
begin
ps2_byte<= ps2_byte_buf; // 锁存当前键值
key_pressed <= 'b1; // 按下标志置一
end
else
key_pressed <= 'b0; // 按下标志清零
end
end
*/
reg[7:0] ps2_asci; //接收数据的相应ASCII码
always @ (ps2_byte_r) begin
case (ps2_byte_r) //键值转换为ASCII码,这里做的比较简单,只处理字母
8'h15: ps2_asci <= 8'h51; //Q
8'h1d: ps2_asci <= 8'h57; //W
8'h24: ps2_asci <= 8'h45; //E
8'h2d: ps2_asci <= 8'h52; //R
8'h2c: ps2_asci <= 8'h54; //T
8'h35: ps2_asci <= 8'h59; //Y
8'h3c: ps2_asci <= 8'h55; //U
8'h43: ps2_asci <= 8'h49; //I
8'h44: ps2_asci <= 8'h4f; //O
8'h4d: ps2_asci <= 8'h50; //P
8'h1c: ps2_asci <= 8'h41; //A
8'h1b: ps2_asci <= 8'h53; //S
8'h23: ps2_asci <= 8'h44; //D
8'h2b: ps2_asci <= 8'h46; //F
8'h34: ps2_asci <= 8'h47; //G
8'h33: ps2_asci <= 8'h48; //H
8'h3b: ps2_asci <= 8'h4a; //J
8'h42: ps2_asci <= 8'h4b; //K
8'h4b: ps2_asci <= 8'h4c; //L
8'h1a: ps2_asci <= 8'h5a; //Z
8'h22: ps2_asci <= 8'h58; //X
8'h21: ps2_asci <= 8'h43; //C
8'h2a: ps2_asci <= 8'h56; //V
8'h32: ps2_asci <= 8'h42; //B
8'h31: ps2_asci <= 8'h4e; //N
8'h3a: ps2_asci <= 8'h4d; //M
default: ;
endcase
end
assign ps2_byte = ps2_asci;
assign ps2_state = ps2_state_r;
//==================keyboard driver part over======================
//=======================1KHz div====display part start===================
parameter N2=50000;
reg clk3=1'b0;
reg [16:0]count3=17'd0;
//assign clk_out=clk3;
always @(posedge clk or negedge rst_n)
begin
if (!rst_n)
begin
count3<=17'd0;
clk3<=1'b0;
end
else
if(count3
顶层模块代码:
`timescale 1ns / 1ps
//////////////////////////////////////////////////////////////////////////////////
// Company:
// Engineer:
//
// Create Date: 10:05:53 03/04/2014
// Design Name:
// Module Name: rs232_send_top
// Project Name:
// Target Devices:
// Tool versions:
// Description:
//
// Dependencies:
//
// Revision:
// Revision 0.01 - File Created
// Additional Comments:
//
//////////////////////////////////////////////////////////////////////////////////
module rs232_send_top(
// tx_start,
//data_byte_in,
tx_data_out,
clk,
rst,
//============
ps2k_clk,
ps2k_data,
sm_bit,
sm_seg,
// ps2_state,
//================
);
input clk;
input rst;
//input [7:0]data_byte_in;
wire tx_start;
output tx_data_out;
//调用模块管脚================
input ps2k_clk;
input ps2k_data;
output [1:0]sm_bit;
output [7:0]sm_seg;
wire ps2_state;
//=====================
wire [7:0]ps2_byte;
//====================
rs_clk_gen M1(
.clk(clk),//
.rst(rst),//
.rs_clk(rs_clk),////
.rs_ena(rs_ena)////
);
rs_send M2(
.tx_start(~ps2_state), //这里采用ps_state ps2模块接收数据有效信号作为发送使能信号
.data_byte_in(ps2_byte),//将ps2接收的1byte信号作为输入
.rs_clk(rs_clk),////
.rs_ena(rs_ena),////
.tx_data_out(tx_data_out),
.clk(clk),//
.rst(rst)//
);
ps2_driever M3(
.clk(clk),
.rst_n(rst),
.ps2k_clk(ps2k_clk),
.ps2k_data(ps2k_data),
.sm_bit(sm_bit),
.sm_seg(sm_seg),
.ps2_state(ps2_state),
.ps2_byte(ps2_byte));
endmodule