当前位置: 首页 > article >正文

[FPGA]Spartan6 Uart固定波特率读写JY901P惯导模块

这版本是固定波特率无法修改串口波特率无法恢复出厂设置出厂设置会更改波特率到9600除非固定波特率一开始设置为9600其他写命令都可以成功写入。1. JY901P交互协议这个是JY901P惯导模块串口的交互协议JY901P串口通讯协议1.1 读格式JY901P传来的数据都是如下格式0x55开头的。默认传出以下四条数据加速度输出、角速度输出、角度输出、磁场输出。分别是0x55 0x51……0x55 0x52……0x55 0x53……0x55 0x54……1.2 写格式写格式较为简单协议头0xFF 0xAA 后面8位寄存器地址再接16位数据位即可。常用的写命令有加速度校准波特率修改数据输出率修改磁场校准恢复出厂设置解锁保存等。2. verilog代码JY901P相关代码结构如下2.1 JY901P_Uart_top模块例化所有JY901P相关模块并对外提供接口timescale 1ns / 1ps module JY901P_Uart_top ( input clk , // 系统时钟50MHz input rst_n , // 全局复位低有效 // 串口物理接口 input uart_rxd , // 串口接收 output uart_txd , // 串口发送 // JY901指令配置接口 input send_cmd_en , // 发送指令使能 input [7:0] cmd_addr , // 指令寄存器地址 input [15:0] cmd_data , // 指令参数 output cmd_send_done , // 指令发送完成 // JY901解析数据输出 output [15:0] acc_x , // X轴加速度 output [15:0] acc_y , output [15:0] acc_z , output [15:0] gyro_x , // X轴角速度 output [15:0] gyro_y , output [15:0] gyro_z , output [15:0] angle_x , // X轴角度 output [15:0] angle_y , output [15:0] angle_z , output [15:0] mag_x , // X轴磁场 output [15:0] mag_y , output [15:0] mag_z , output [63:0] JY901P_data_out , output frame_valid // 完整帧接收完成标志 ); // 内部连线 wire [7:0] uart_rx_data; // UART接收数据 wire uart_rx_flag; // UART接收标志 wire [7:0] uart_tx_data; // UART发送数据 wire uart_tx_flag; // UART发送标志 wire uart_tx_done; // UART单字节发送完成 // 例化UART接收模块 uart_recceive #( .UART_BPS(115200), .CLK_FREQ(50000000) ) uart_RX ( .clk (clk) , .sys_rst_n (rst_n) , .rx (uart_rxd) , .po_data (uart_rx_data), .po_flag (uart_rx_flag) ); // 例化JY901顶层模块 JY901P_top #( .CLK_FREQ(50000000) ) JY901P_top_inst ( .clk (clk) , .rst_n (rst_n) , // 写命令 .uart_tx_data (uart_tx_data) , .uart_tx_flag (uart_tx_flag) , .uart_tx_done (uart_tx_done) , .send_cmd_en (send_cmd_en) , .cmd_addr (cmd_addr) , .cmd_data (cmd_data) , .cmd_send_done (cmd_send_done), // 读数据 .uart_rx_data (uart_rx_data) , .uart_rx_flag (uart_rx_flag) , .acc_x (acc_x) , .acc_y (acc_y) , .acc_z (acc_z) , .gyro_x (gyro_x) , .gyro_y (gyro_y) , .gyro_z (gyro_z) , .angle_x (angle_x) , .angle_y (angle_y) , .angle_z (angle_z) , .mag_x (mag_x) , .mag_y (mag_y) , .mag_z (mag_z) , .JY901P_data_out(JY901P_data_out), .frame_valid (frame_valid) ); // 例化UART发送模块 uart_send #( .UART_BPS(115200), .CLK_FREQ(50000000) ) uart_TX ( .clk (clk) , .sys_rst_n (rst_n) , .pi_data (uart_tx_data), .pi_flag (uart_tx_flag), .tx (uart_txd) , .tx_done (uart_tx_done) ); endmodule2.2 JY901P_top模块例化JY901P_reader和JY901P_writer对外输出解析后的惯导数据module JY901P_top #( parameter CLK_FREQ 50_000_000 // 系统时钟50MHz ) ( input clk , // 系统时钟50MHz input rst_n , // 全局复位低有效 // UART接口 input [7:0] uart_rx_data , // UART接收数据 input uart_rx_flag , // UART接收标志 output reg [7:0] uart_tx_data , // UART发送数据 output reg uart_tx_flag , // UART发送标志 input uart_tx_done , // UART单字节发送完成 // 指令配置接口向JY901P发命令 input send_cmd_en , // 发送指令使能 input [7:0] cmd_addr , // 指令寄存器地址 input [15:0] cmd_data , // 指令参数 output reg cmd_send_done , // 指令发送完成 // 解析后的数据输出给其他模块 output reg [15:0] acc_x , // X轴加速度 output reg [15:0] acc_y , output reg [15:0] acc_z , output reg [15:0] gyro_x , // X轴角速度 output reg [15:0] gyro_y , output reg [15:0] gyro_z , output reg [15:0] angle_x , // X轴角度 output reg [15:0] angle_y , output reg [15:0] angle_z , output reg [15:0] mag_x , // X轴磁场 output reg [15:0] mag_y , output reg [15:0] mag_z , output [63:0] JY901P_data_out , output reg frame_valid // 同一组完整帧接收完成标志 ); // 内部连线 // 读模块输出连线 wire [15:0] acc_x_r, acc_y_r, acc_z_r; wire [15:0] gyro_x_r, gyro_y_r, gyro_z_r; wire [15:0] angle_x_r, angle_y_r, angle_z_r; wire [15:0] mag_x_r, mag_y_r, mag_z_r; wire frame_valid_r; // 写模块输出连线 wire [7:0] uart_tx_data_w; wire uart_tx_flag_w; wire cmd_send_done_w; // 例化JY901读模块 JY901P_reader #( .CLK_FREQ(CLK_FREQ) ) JY901P_reader_inst ( .clk (clk) , .rst_n (rst_n) , .uart_rx_data (uart_rx_data) , .uart_rx_flag (uart_rx_flag) , .acc_x (acc_x_r) , .acc_y (acc_y_r) , .acc_z (acc_z_r) , .gyro_x (gyro_x_r) , .gyro_y (gyro_y_r) , .gyro_z (gyro_z_r) , .angle_x (angle_x_r) , .angle_y (angle_y_r) , .angle_z (angle_z_r) , .mag_x (mag_x_r) , .mag_y (mag_y_r) , .mag_z (mag_z_r) , .JY901P_data_out(JY901P_data_out), .frame_valid (frame_valid_r) ); // 例化JY901写模块 JY901P_writer #( .CLK_FREQ(CLK_FREQ) ) JY901P_writer_inst ( .clk (clk) , .rst_n (rst_n) , .send_cmd_en (send_cmd_en) , .cmd_addr (cmd_addr) , .cmd_data (cmd_data) , .uart_tx_data (uart_tx_data_w), .uart_tx_flag (uart_tx_flag_w), .uart_tx_done (uart_tx_done) , .cmd_send_done (cmd_send_done_w) ); // 输出赋值 always (posedge clk or negedge rst_n) begin if(!rst_n) begin // 数据输出复位 acc_x 16d0; acc_y 16d0; acc_z 16d0; gyro_x 16d0; gyro_y 16d0; gyro_z 16d0; angle_x 16d0; angle_y 16d0; angle_z 16d0; mag_x 16d0; mag_y 16d0; mag_z 16d0; frame_valid 1b0; // UART发送接口复位 uart_tx_data 8h00; uart_tx_flag 1b0; cmd_send_done 1b0; end else begin // 转发读模块数据 acc_x acc_x_r; acc_y acc_y_r; acc_z acc_z_r; gyro_x gyro_x_r; gyro_y gyro_y_r; gyro_z gyro_z_r; angle_x angle_x_r; angle_y angle_y_r; angle_z angle_z_r; mag_x mag_x_r; mag_y mag_y_r; mag_z mag_z_r; frame_valid frame_valid_r; // 转发写模块UART数据 uart_tx_data uart_tx_data_w; uart_tx_flag uart_tx_flag_w; cmd_send_done cmd_send_done_w; end end endmodule2.3 JY901P_reader模块检测帧头0x55逐字节接收并计算和校验区分0x51/0x52/0x53/0x54四种数据帧严格按照数据顺序接收只有完整一组才输出有效信号解析出16位有符号数加速度角速度角度磁场输出有效帧脉冲frame_valid单帧接收状态机IDLE→收类型→收数据→校验帧顺序状态机等待 ACC→等待 GYRO→等待 ANGLE→等待 MAG→完成只有顺序完全正确才输出有效数据从根源上避免乱帧、丢帧问题。module JY901P_reader #( parameter CLK_FREQ 50_000_000 ) ( input clk, input rst_n, input [7:0] uart_rx_data, input uart_rx_flag, output reg [15:0] acc_x, output reg [15:0] acc_y, output reg [15:0] acc_z, output reg [15:0] gyro_x, output reg [15:0] gyro_y, output reg [15:0] gyro_z, output reg [15:0] angle_x, output reg [15:0] angle_y, output reg [15:0] angle_z, output reg [15:0] mag_x, output reg [15:0] mag_y, output reg [15:0] mag_z, output reg [63:0] JY901P_data_out, output reg frame_valid ); // 协议常量定义 localparam FRAME_HEAD 8h55; localparam DATA_TYPE_ACC 8h51; localparam DATA_TYPE_GYRO 8h52; localparam DATA_TYPE_ANGLE 8h53; localparam DATA_TYPE_MAG 8h54; // 单帧接收状态 localparam S_IDLE 4d0; localparam S_RECV_TYPE 4d1; localparam S_RECV_DATA 4d2; localparam S_RECV_CRC 4d3; // 帧顺序状态 localparam S_WAIT_ACC 4d4; localparam S_WAIT_GYRO 4d5; localparam S_WAIT_ANGLE 4d6; localparam S_WAIT_MAG 4d7; localparam S_DONE 4d8; // 内部信号 reg [3:0] curr_state; reg [3:0] next_state; reg [3:0] seq_state; reg [3:0] byte_cnt; reg [7:0] frame_buf [0:10]; reg [7:0] data_type; reg [7:0] check_sum; reg rx_flag_d1; reg rx_flag_d2; wire rx_pulse; reg [15:0] acc_x_buf, acc_y_buf, acc_z_buf; reg [15:0] gyro_x_buf, gyro_y_buf, gyro_z_buf; reg [15:0] ang_x_buf, ang_y_buf, ang_z_buf; reg [15:0] mag_x_buf, mag_y_buf, mag_z_buf; // 上升沿捕获 assign rx_pulse rx_flag_d1 ~rx_flag_d2; // 同步打拍 always (posedge clk or negedge rst_n) begin if(!rst_n) {rx_flag_d2, rx_flag_d1} 2b00; else {rx_flag_d2, rx_flag_d1} {rx_flag_d1, uart_rx_flag}; end // // 第一段时序逻辑 —— 状态更新 // always (posedge clk or negedge rst_n) begin if(!rst_n) curr_state S_IDLE; else curr_state next_state; end // // 第二段组合逻辑 —— 状态跳转 // always (*) begin next_state curr_state; case(curr_state) S_IDLE: if(rx_pulse uart_rx_data FRAME_HEAD) next_state S_RECV_TYPE; S_RECV_TYPE: if(rx_pulse) next_state S_RECV_DATA; S_RECV_DATA: if(rx_pulse byte_cnt 9) next_state S_RECV_CRC; S_RECV_CRC: if(rx_pulse) next_state S_IDLE; endcase end // // 第三段时序逻辑 —— Moore 输出数据接收 解析 // always (posedge clk or negedge rst_n) begin if(!rst_n) begin byte_cnt 0; check_sum 0; data_type 0; seq_state S_WAIT_ACC; frame_valid 0; acc_x 0; acc_y 0; acc_z 0; gyro_x 0; gyro_y 0; gyro_z 0; angle_x 0; angle_y 0; angle_z 0; mag_x 0; mag_y 0; mag_z 0; JY901P_data_out 0; end else begin frame_valid 0; if(rx_pulse) begin case(curr_state) S_IDLE: begin byte_cnt 1; check_sum uart_rx_data; end S_RECV_TYPE: begin data_type uart_rx_data; check_sum check_sum uart_rx_data; byte_cnt byte_cnt 1; end S_RECV_DATA: begin frame_buf[byte_cnt] uart_rx_data; check_sum check_sum uart_rx_data; byte_cnt byte_cnt 1; end S_RECV_CRC: begin byte_cnt 0; if(uart_rx_data check_sum) begin case(data_type) DATA_TYPE_ACC: begin acc_x_buf {frame_buf[3],frame_buf[2]}; acc_y_buf {frame_buf[5],frame_buf[4]}; acc_z_buf {frame_buf[7],frame_buf[6]}; end DATA_TYPE_GYRO: begin gyro_x_buf {frame_buf[3],frame_buf[2]}; gyro_y_buf {frame_buf[5],frame_buf[4]}; gyro_z_buf {frame_buf[7],frame_buf[6]}; end DATA_TYPE_ANGLE: begin ang_x_buf {frame_buf[3],frame_buf[2]}; ang_y_buf {frame_buf[5],frame_buf[4]}; ang_z_buf {frame_buf[7],frame_buf[6]}; end DATA_TYPE_MAG: begin mag_x_buf {frame_buf[3],frame_buf[2]}; mag_y_buf {frame_buf[5],frame_buf[4]}; mag_z_buf {frame_buf[7],frame_buf[6]}; end endcase case(seq_state) S_WAIT_ACC: seq_state (data_typeDATA_TYPE_ACC) ? S_WAIT_GYRO : S_WAIT_ACC; S_WAIT_GYRO: seq_state (data_typeDATA_TYPE_GYRO) ? S_WAIT_ANGLE : S_WAIT_ACC; S_WAIT_ANGLE: seq_state (data_typeDATA_TYPE_ANGLE)? S_WAIT_MAG : S_WAIT_ACC; S_WAIT_MAG: seq_state (data_typeDATA_TYPE_MAG) ? S_DONE : S_WAIT_ACC; endcase end end endcase end if(seq_state S_DONE) begin acc_x acc_x_buf; acc_y acc_y_buf; acc_z acc_z_buf; gyro_x gyro_x_buf; gyro_y gyro_y_buf; gyro_z gyro_z_buf; angle_x ang_x_buf; angle_y ang_y_buf; angle_z ang_z_buf; mag_x mag_x_buf; mag_y mag_y_buf; mag_z mag_z_buf; JY901P_data_out {8h55,8h51,ang_x_buf,ang_y_buf,ang_z_buf}; frame_valid 1; seq_state S_WAIT_ACC; end end end endmodule2.4 JY901P_writer模块接收外部指令使能地址数据拼接5字节指令帧FF AA ADDR DATAH DATAL状态机控制逐字节发送等待UART发送完成应答tx_done全部发送完成输出cmd_send_doneIDLE等待发送触发SEND_BYTE发送当前字节WAIT_DONE等待 UART 单字节发送完成发送流程完全自动化外部只需要给一个单周期脉冲即可启动配置。module JY901P_writer #( parameter CLK_FREQ 50_000_000 // 系统时钟50MHz ) ( input clk , // 系统时钟 input rst_n , // 低电平复位 // 指令配置接口 input send_cmd_en , // 发送指令使能单周期脉冲 input [7:0] cmd_addr , // 寄存器地址 input [15:0] cmd_data , // 配置数据 // UART 交互接口 output reg [7:0] uart_tx_data , // UART发送数据 output reg uart_tx_flag , // UART发送标志 input uart_tx_done , // UART单字节发送完成 output reg cmd_send_done // 整条指令发送完成 ); // 指令帧定义 localparam CMD_HEAD1 8hFF; // 帧头1 localparam CMD_HEAD2 8hAA; // 帧头2 localparam CMD_FRAME_LEN 5; // 总字节数5字节 // 状态定义 localparam S_IDLE 3d0; // 空闲 localparam S_SEND_H1 3d1; // 发送 0xFF localparam S_SEND_H2 3d2; // 发送 0xAA localparam S_SEND_ADDR 3d3; // 发送地址 localparam S_SEND_DH 3d4; // 发送数据高字节 localparam S_SEND_DL 3d5; // 发送数据低字节 localparam S_DONE 3d6; // 发送完成 // 内部信号 reg [2:0] current_state; reg [2:0] next_state; reg [2:0] byte_cnt; // 字节计数 // // 第一段时序逻辑 —— 现态更新 // always (posedge clk or negedge rst_n) begin if(!rst_n) current_state S_IDLE; else current_state next_state; end // // 第二段组合逻辑 —— 次态跳转 // always (*) begin next_state current_state; case(current_state) S_IDLE: begin if(send_cmd_en) next_state S_SEND_H1; end S_SEND_H1: if(uart_tx_done) next_state S_SEND_H2; S_SEND_H2: if(uart_tx_done) next_state S_SEND_ADDR; S_SEND_ADDR: if(uart_tx_done) next_state S_SEND_DH; S_SEND_DH: if(uart_tx_done) next_state S_SEND_DL; S_SEND_DL: if(uart_tx_done) next_state S_DONE; S_DONE: next_state S_IDLE; default: next_state S_IDLE; endcase end // // 第三段时序逻辑 —— Moore 输出只与当前状态有关 // always (posedge clk or negedge rst_n) begin if(!rst_n) begin uart_tx_data 8d0; uart_tx_flag 1b0; cmd_send_done 1b0; end else begin // 默认值 uart_tx_flag 1b0; cmd_send_done 1b0; case(current_state) S_SEND_H1: begin uart_tx_data CMD_HEAD1; uart_tx_flag 1b1; end S_SEND_H2: begin uart_tx_data CMD_HEAD2; uart_tx_flag 1b1; end S_SEND_ADDR: begin uart_tx_data cmd_addr; uart_tx_flag 1b1; end S_SEND_DH: begin uart_tx_data cmd_data[15:8]; uart_tx_flag 1b1; end S_SEND_DL: begin uart_tx_data cmd_data[7:0]; uart_tx_flag 1b1; end S_DONE: begin cmd_send_done 1b1; end endcase end end endmodule2.5 uart_send模块并行数据转串口发送发送完成后输出tx_done应答信号严格遵循UART协议起始位8位数据停止位module uart_send #( parameter UART_BPS 115200, // 串口波特率 parameter CLK_FREQ 50000000 // 系统时钟频率 ) ( input clk, // 系统时钟 input sys_rst_n, // 复位信号低有效 input [7:0] pi_data, // 待发送的1字节数据 input pi_flag, // 发送触发信号1个时钟周期高电平 output reg tx, // 串口发送数据 output reg tx_done // 1字节发送完成信号1个时钟周期高电平 ); // 本地参数定义 localparam BAUD_CNT_MAX CLK_FREQ / UART_BPS; // 波特率计数最大值 // 寄存器定义 reg [15:0] baud_cnt; // 波特率计数器 reg [3:0] bit_cnt; // 字节内位数计数器0~9起始位8数据位停止位 reg [7:0] data_buf; // 待发送数据缓存 reg tx_en; // 发送使能信号 reg pi_flag_d1; // pi_flag打拍避免漏采 // 提取pi_flag上升沿确保1周期的pi_flag被捕获 wire pi_flag_posedge pi_flag ~pi_flag_d1; // 步骤1pi_flag打拍 捕获上升沿锁存数据 always (posedge clk or negedge sys_rst_n) begin if (!sys_rst_n) begin data_buf 8d0; tx_en 1b0; pi_flag_d1 1b0; end else begin pi_flag_d1 pi_flag; // 打拍 if (pi_flag_posedge) begin // 捕获上升沿避免漏触发 data_buf pi_data; tx_en 1b1; $display([%0t] uart_send捕获到触发信号锁存数据0x%02X, $time, pi_data); end else if (bit_cnt 4d9 baud_cnt BAUD_CNT_MAX - 1) begin // 修复停止位发送完成后再关闭tx_en tx_en 1b0; $display([%0t] uart_send关闭发送使能, $time); end end end // 步骤2波特率计数器计时1个位周期 always (posedge clk or negedge sys_rst_n) begin if (!sys_rst_n) begin baud_cnt 16d0; end else if (tx_en) begin // 发送使能时计数 if (baud_cnt BAUD_CNT_MAX - 1) begin baud_cnt 16d0; end else begin baud_cnt baud_cnt 1b1; end end else begin baud_cnt 16d0; end end // 步骤3字节内位数计数器计数1字节的10个位 always (posedge clk or negedge sys_rst_n) begin if (!sys_rst_n) begin bit_cnt 4d0; end else if (tx_en) begin if (baud_cnt BAUD_CNT_MAX - 1) begin // 1个位周期完成 if (bit_cnt 4d9) begin bit_cnt 4d0; end else begin bit_cnt bit_cnt 1b1; end end end else begin bit_cnt 4d0; end end // 步骤4串口数据发送按位输出 always (posedge clk or negedge sys_rst_n) begin if (!sys_rst_n) begin tx 1b1; // 串口空闲时为高电平 end else if (tx_en) begin case (bit_cnt) 4d0: tx 1b0; // 起始位低电平 4d1: tx data_buf[0]; // 数据位bit0 4d2: tx data_buf[1]; // 数据位bit1 4d3: tx data_buf[2]; // 数据位bit2 4d4: tx data_buf[3]; // 数据位bit3 4d5: tx data_buf[4]; // 数据位bit4 4d6: tx data_buf[5]; // 数据位bit5 4d7: tx data_buf[6]; // 数据位bit6 4d8: tx data_buf[7]; // 数据位bit7 4d9: tx 1b1; // 停止位高电平 default: tx 1b1; endcase end else begin tx 1b1; end end // 步骤5发送完成信号tx_done生成1个时钟周期高电平 always (posedge clk or negedge sys_rst_n) begin if (!sys_rst_n) begin tx_done 1b0; end else if (tx_en (bit_cnt 4d9) (baud_cnt BAUD_CNT_MAX - 1)) begin tx_done 1b1; // 1字节发送完成拉高1周期 $display([%0t] uart_send1字节发送完成tx_done拉高, $time); end else begin tx_done 1b0; end end endmodule2.6 uart_receive模块起始位下降沿检测比特位中间时刻采样保证数据稳定接收完成后输出单周期脉冲标志po_flagmodule uart_recceive #( parameter UART_BPS d115200, //串口波特率 parameter CLK_FREQ d50_000_000 //时钟频率 ) ( input wire clk , //系统时钟100MHz input wire sys_rst_n , //全局复位 input wire rx , //串口接收数据 output reg [7:0] po_data , //串转并后的8bit数据 output reg po_flag //串转并后的数据有效标志信号 ); localparam BAUD_CNT_MAX CLK_FREQ/UART_BPS ; //reg define reg rx_reg1 ; reg rx_reg2 ; reg rx_reg3 ; reg start_nedge ; reg work_en ; reg [12:0] baud_cnt ; reg bit_flag ; reg [3:0] bit_cnt ; reg [7:0] rx_data ; reg rx_flag ; //接收完成标志位 //插入两级寄存器进行数据同步用来消除亚稳态 //rx_reg1:第一级寄存器寄存器空闲状态复位为1 always(posedge clk or negedge sys_rst_n) if(sys_rst_n 1b0) rx_reg1 1b1; else rx_reg1 rx; //rx_reg2:第二级寄存器寄存器空闲状态复位为1 always(posedge clk or negedge sys_rst_n) if(sys_rst_n 1b0) rx_reg2 1b1; else rx_reg2 rx_reg1; //rx_reg3:第三级寄存器和第二级寄存器共同构成下降沿检测 always(posedge clk or negedge sys_rst_n) if(sys_rst_n 1b0) rx_reg3 1b1; else rx_reg3 rx_reg2; //start_nedge:检测到下降沿时start_nedge产生一个时钟的高电平 always(posedge clk or negedge sys_rst_n) if(sys_rst_n 1b0) start_nedge 1b0; else if((~rx_reg2) (rx_reg3)) start_nedge 1b1; else start_nedge 1b0; //work_en:接收数据工作使能信号 always(posedge clk or negedge sys_rst_n) if(sys_rst_n 1b0) work_en 1b0; else if(start_nedge 1b1) work_en 1b1; else if((bit_cnt 4d8) (bit_flag 1b1)) work_en 1b0; //baud_cnt:波特率计数器计数从0计数到BAUD_CNT_MAX - 1 always(posedge clk or negedge sys_rst_n) if(sys_rst_n 1b0) baud_cnt 13b0; else if((baud_cnt BAUD_CNT_MAX - 1) || (work_en 1b0)) baud_cnt 13b0; else if(work_en 1b1) baud_cnt baud_cnt 1b1; //bit_flag:当baud_cnt计数器计数到中间数时采样的数据最稳定 //此时拉高一个标志信号表示数据可以被取走 always(posedge clk or negedge sys_rst_n) if(sys_rst_n 1b0) bit_flag 1b0; else if(baud_cnt BAUD_CNT_MAX/2 - 1) bit_flag 1b1; else bit_flag 1b0; //bit_cnt:有效数据个数计数器当8个有效数据不含起始位和停止位 //都接收完成后计数器清零 always(posedge clk or negedge sys_rst_n) if(sys_rst_n 1b0) bit_cnt 4b0; else if((bit_cnt 4d8) (bit_flag 1b1)) bit_cnt 4b0; else if(bit_flag 1b1) bit_cnt bit_cnt 1b1; //rx_data:输入数据进行移位 always(posedge clk or negedge sys_rst_n) if(sys_rst_n 1b0) rx_data 8b0; else if((bit_cnt 4d1)(bit_cnt 4d8)(bit_flag 1b1)) rx_data {rx_reg3, rx_data[7:1]}; //rx_flag:输入数据移位完成时rx_flag拉高一个时钟的高电平 always(posedge clk or negedge sys_rst_n) if(sys_rst_n 1b0) rx_flag 1b0; else if((bit_cnt 4d8) (bit_flag 1b1)) rx_flag 1b1; //移位完成拉高rx_flag电平 else rx_flag 1b0; //po_data:输出完整的8位有效数据 比rx_data延后一个时钟周期 always(posedge clk or negedge sys_rst_n) if(sys_rst_n 1b0) po_data 8b0; else if(rx_flag 1b1) po_data rx_data; //po_flag:输出数据有效标志比rx_flag延后一个时钟周期为了和po_data同步 always(posedge clk or negedge sys_rst_n) if(sys_rst_n 1b0) po_flag 1b0; else po_flag rx_flag; endmodule

相关文章:

[FPGA]Spartan6 Uart固定波特率读写JY901P惯导模块

这版本是固定波特率,无法修改串口波特率,无法恢复出厂设置(出厂设置会更改波特率到9600,除非固定波特率一开始设置为9600,其他写命令都可以成功写入)。 1. JY901P交互协议 这个是JY901P惯导模块串口的交互…...

tiktok最新V2滑块验证分析 /captcha/verifyV2

经过好些天的努力,终于攻克了tiktokV2滑块,踩过不少坑,今天来总结一下.首先,通过/captcha/get获取验证信息,获取到的信息中,除了背景前景图片url外,challenge_id ,tip_y 这两个都是有用的.然后,我们来看看要提交给/captcha/verifyV2的数据:最主要的有reply,mm,mp,tmv,gy 这几个…...

2026年OpenClaw怎么集成?阿里云1分钟保姆级教程+大模型APIKey配置、Skill集成教程

2026年OpenClaw怎么集成?阿里云1分钟保姆级教程大模型APIKey配置、Skill集成教程。本文面向零基础用户,完整说明在轻量服务器与本地Windows11、macOS、Linux系统中部署OpenClaw(Clawdbot)的流程,包含环境配置、服务启动…...

第5章,[标签 Win32] :GDI 的其他方面的分类

专栏导航 上一篇:第5章,[标签 Win32] :GDI 的基本图形 回到目录 下一篇:无 本节前言 对于本节所讲解的知识,有可能,你会需要时不时地参考本专栏的其它文章。真的遇到了需要参考之前的文章的知识点&…...

收藏!小白/程序员入行AI应用开发必看,别被招聘要求吓退(附实操资源)

如果你是程序员小白,或是想转型AI应用开发的从业者,听我一句劝——大胆投简历,别被招聘启事上的“精通大模型底层原理”“2年以上AI相关经验”吓住!很多时候,招聘要求写的只是企业的“理想画像”,我和身边不…...

第5章,[标签 Win32] :GDI 的基本图形

专栏导航 上一篇:第5章,[标签 Win32] :GDI 函数调用 回到目录 下一篇:第5章,[标签 Win32] :GDI 的其他方面的分类 本节前言 对于本节所讲解的知识,有可能,你会需要时不时地参考…...

Day05:C语言数组存储结构与字符串详解

一、数组的存储结构1. 数组变量的地址连续性数组中的元素在内存中地址是连续的。数组名非常重要,涉及指针与内存操作。2. 数组名的含义数组名表示首元素的地址。示例:int arr[5]; printf("%p\n", arr); // 输出首元素地址 printf("%p…...

pgRouting安装及使用示例

文章目录环境文档用途详细信息环境 系统平台:Linux x86-64 Red Hat Enterprise Linux 7 版本:4.5.10 文档用途 本文介绍pgRouting的安装及使用示例。 详细信息 简介 pgRouting是PostgreSQL下基于PostGIS的扩展插件,提供了地理空间路由和…...

day02统计师考试(初级)统计法的特点

统计法的特点 (一)调整对象具有特殊性和复杂性 1.调整对象的特殊性: 统计法以统计活动中形成的社会关系为调整对象。 2.调整对象的复杂性: ①调整的社会关系既有纵向的管理关系,也有横向的指导关系; ②既有…...

数据库无法连接情况排查

文章目录环境症状问题原因解决方案环境 系统平台:N/A 版本:9.0,6.0,4.5 症状 本文档用于提供HGDB数据库的常见无法连接问题的基本排查思路,建议按顺序排查; 若以上步骤未能排查出连接问题,建议联系瀚高厂家处理。 …...

一文讲透数字化转型的十个关键概念:信息化、自动化、数据化、智能化、平台化……

最近几年,提到数字化转型,总绕不开一堆带“化”的词:信息化、数据化、智能化、平台化等等。说实话,这些概念太多了,有时候连从业者都容易搞混。今天我就来给大家梳理一下电子化、信息化、结构化、多媒体化、自动化、网…...

开源TOP20项目(2026.04.01-2026.04.06)

排名项目名Star描述1luongnv89/claude-howto20.2kClaude Code 的可视化、示例驱动指南——从基本概念到高级代理,提供可立即产生价值的复制粘贴模板。从打字claude到编排代理、钩子、技能和 MCP 服务器——通过可视化教程、复制粘贴模板和引导式学习路径2NousResear…...

通义千问2.5-7B低成本上线:共享GPU资源部署案例

通义千问2.5-7B低成本上线:共享GPU资源部署案例 想体验最新最强的开源大模型,但被动辄几十GB的显存需求和昂贵的专业显卡劝退?这可能是很多开发者和创业团队面临的现实困境。今天,我们就来分享一个极具性价比的解决方案&#xff…...

反思学习!

前言之前挖的小程序,没找到漏洞,挖的web没找到漏洞,然后这次买了fofa会员,不买应该也能挖到这次的侧重点不一样了,以前学校的首页啊,什么学院啊,我都能看半天,看着看着就知道了&…...

从图像压缩到信道反馈:CsiNet如何重塑大规模MIMO的深度学习范式

1. 当无线通信遇上计算机视觉:CSI为何能被看作图像? 第一次听说把信道状态信息(CSI)当作图像处理时,我的反应和大多数通信工程师一样:"这脑洞开得有点大吧?"但当我真正动手复现CsiNet…...

20个核心AI概念轻松入门:收藏这份小白友好指南,开启大模型学习之旅!

如果你曾尝试学习AI,大概率至少有过一次这样的感受……“这到底在讲什么?” 术语太多。 工具太多。 网上所有人都说得好像理所当然。 学习AI很容易让人感到崩溃。 尤其如果你不是直接从事这一行,几乎像在学一门全新的语言。 但我逐渐意识到一…...

工业仿真混合引擎实时调度策略解析

工业场景下,混合引擎(通常指融合了传统物理求解器与AI/ML代理模型或神经求解器的仿真系统)的实时调度策略是实现数字孪生、预测性维护和实时优化的核心技术瓶颈。其核心目标是在满足确定性延迟和计算精度的前提下,动态分配计算资源…...

AWS 账单查看与付款方式设置指南(企业支持实用手册)

一文搞定 AWS 发票下载、费用明细查询和电汇付款配置,适合企业财务和运维人员快速上手。 前言 使用 AWS 的企业经常会遇到这几个问题:月底了发票在哪下载?费用明细怎么导出给财务?公司要用银行电汇付款怎么设置? 这篇文章把这三件事讲清楚,都是控制台操作,不需要写代码…...

Glyph视觉推理新手必看:如何用智谱开源模型轻松处理超长合同与论文

Glyph视觉推理新手必看:如何用智谱开源模型轻松处理超长合同与论文 1. 从痛点出发:为什么你需要Glyph? 想象一下,你手头有一份长达200页的合同,或者一篇包含复杂图表和公式的学术论文。你需要快速找到关键条款&#…...

从零学卷积神经网络——梯度下降,反向传播,卷积核权重视觉对比

很多人在刚接触卷积神经网络时,会被满屏的矩阵数字搞晕。其实,卷积核并不是冰冷的算式,你可以把它想象成一副副“神奇眼镜”。比如这张 77 的图像,左上和右下是亮区,其他地方是暗区。现在,我们让它分别戴上…...

Pixel Language Portal 助力后端开发:构建高并发实时数据处理服务

Pixel Language Portal 助力后端开发:构建高并发实时数据处理服务 1. 实时数据处理的行业痛点 想象一下这样的场景:一家智能工厂部署了上千个传感器,每秒产生数百万条数据;或者一个金融交易平台,需要实时处理全球市场…...

如何快速掌握WandEnhancer使用:面向新手的完整免费增强指南

如何快速掌握WandEnhancer使用:面向新手的完整免费增强指南 【免费下载链接】Wand-Enhancer Advanced UX and interoperability extension for Wand (WeMod) app 项目地址: https://gitcode.com/gh_mirrors/we/Wand-Enhancer WandEnhancer是一款专为游戏辅助…...

2025年终极指南:R3nzSkin国服特供版——一键解锁LOL全皮肤的完整解决方案

2025年终极指南:R3nzSkin国服特供版——一键解锁LOL全皮肤的完整解决方案 【免费下载链接】R3nzSkin-For-China-Server Skin changer for League of Legends (LOL) 项目地址: https://gitcode.com/gh_mirrors/r3/R3nzSkin-For-China-Server 你是否厌倦了每次…...

100G SFP光模块全解读:核心定义、关键特性与主流应用场景

在高速光通信网络飞速发展的当下,100G速率已成为数据中心、城域网、5G承载网等场景的核心传输需求,而100G SFP光模块作为实现光电信号转换的关键器件,凭借小巧的体积、灵活的适配性,成为连接网络设备、支撑高速数据传输的核心载体…...

当AI搜索引擎开始替用户做消费决策,品牌的媒介宣发逻辑也正在被彻底改写

去年年底,联合利华CEO在内部会上说了句话,传出来后不少品牌人都在转。他说“懒惰营销的时代已经结束了”,一年只拍几条广告、围绕几个新品做营销的传统打法,已经彻底失效。这话放在2026年的媒介宣发语境下,几乎是一份判…...

常州装修设计领域评测与推荐——聚焦实力标杆,认准鸿鹄领跑优势

一、核心引导问题1. 面对常州装修设计行业的趋势,不同规模的企业应如何筛选技术扎实、效果可视的常州装修设计服务商?2. 常州鸿鹄装饰设计工程有限公司凭借哪些核心优势,成功跻身行业头部阵营?3. 常州装修设计行业其核心包含哪些能…...

马尔可夫性、极小性和忠实性的关系:因果图与数据的深层逻辑

马尔可夫性、极小性和忠实性的关系:因果图与数据的深层逻辑 在因果推断中,我们试图通过观测数据来还原背后的因果图(DAG)。然而,图结构与概率分布之间的关系并非绝对的一一对应。为了从数据中锁定唯一的因果结构&#…...

外汇流动性和市场情绪指标MT4、MT5

使用外汇流动性指标交易 外汇流动性指标通过帮助识别关键市场水平来支持贸易规划,包括: 支撑与阻力位 –根据交易密度显著或反复反应的区域确定。供需区——通过被称为买方和卖方流动性区的区域突出显示,这些区域暗示了可能存在未成交的买卖…...

Redis 常用数据类型

下面给你一套面试最标准、逻辑清晰、直接背诵的版本: Redis 常用数据类型 使用场景 底层原理 面试话术,一次性讲全。 一、开场一句话(必说) Redis 是基于内存的高性能 KV 数据库,支持丰富的数据结构,通过…...

【无标题】第二章 Hadoop3安装

2.1 启动Docker容器2.1.1 加载镜像用来将一个Docker镜像从/cg/images/hadoop_node.tar.gz压缩包加载到本地Docker环境里面docker load < /cg/images/hadoop_node.tar.gz运行结果如下&#xff1a;docker run --name master --privileged --ulimit nofile65535:65535 --hostna…...