verilog rs232串口模块

news2025/1/13 0:20:47

前面发了个发送模块,这次补齐,完整。
串口计数器,波特率适配
uart_clk.v

module uart_clk(
    input wire clk,
    input wire rst_n,
    input wire tx_clk_en,
    input wire rx_clk_en,
    input wire[1:0] baud_sel,
    output wire tx_clk,
    output wire rx_clk
);
localparam OSC = 50_000_000;
reg[16:0] bauds[0:3] ;

reg[16:0] tx_cnt;
reg[16:0] rx_cnt;

initial begin
    bauds[0] <= OSC/19200 - 1;
    bauds[1] <= OSC/38400 - 1;
    bauds[2] <= OSC/57600 - 1;
    bauds[3] <= OSC/115200 - 1;
end

always @(posedge clk or negedge rst_n ) begin
    if (!rst_n )
        tx_cnt <= 0;
    else if (!tx_clk_en || tx_cnt === bauds[baud_sel])
        tx_cnt <= 0;
    else
        tx_cnt <= tx_cnt + 1;
end

assign tx_clk = (tx_cnt  === bauds[baud_sel]) ? 1'b1 : 1'b0;

always @(posedge clk or negedge rst_n) begin
    if (!rst_n)
        rx_cnt <= 0;
    else if (!rx_clk_en || rx_cnt === bauds[baud_sel])
        rx_cnt <= 0;
    else
        rx_cnt <= rx_cnt + 1;
end

assign  rx_clk = (rx_cnt === (bauds[baud_sel] >>1)) ? 1'b1 : 1'b0;
    
endmodule

发送模块
tx.v

module tx (
    input wire clk,
    input wire rst_n,
    input wire tx_en,
    input wire tx_clk,
    input wire[7:0] tx_data,
    output reg tx_clk_en,
    output reg txd,
    output reg busy,
    output reg done
);

reg [1:0] stage;
reg [2:0] curbit;
reg [7:0] tx_rdata;

always @(posedge clk or negedge rst_n) begin
    if (!rst_n) begin
        busy <= 1'b0;
    end else if (tx_en && !busy) begin
        busy <= 1'b1;
        tx_rdata <= tx_data;
    end else if (done)
        busy <= 1'b0;
end

always @(busy) begin
    if(busy)
        tx_clk_en <= 1'b1;
    else 
        tx_clk_en <= 1'b0;
end

always @(posedge clk or negedge rst_n) begin
    if (!rst_n) begin
        txd <= 1'b1;
        done <= 1'b0;
        stage <= 2'd0;
    end else if (busy) begin
        if (tx_clk) begin
            if(stage === 2'd0) begin
                txd <= 1'b0;
                done <= 1'b0;
                curbit <= 3'd0;
                stage <= 2'b1;
            end else if (stage === 2'b1) begin
                txd <= tx_rdata[curbit];
                if(curbit == 3'd7)
                    stage <= 2'd2;
                else
                    curbit <= curbit + 1;
            end else if (stage == 2'd2) begin
                txd <= 1'b1;
                stage <= stage + 1;
            end else if (stage == 2'd3) begin
                txd <= 1'b1;
                done <= 1'b1;
                stage <= 2'd0;
            end
        end
    end else begin
        txd <= 1'b1;
        done <= 1'b0;
        stage <= 2'd0;
    end
end

endmodule

接收模块

rx.v

module rx (
    input wire clk,
    input wire rst_n,
    input wire rx_en,
    input wire rx_clk,
    input wire rxd,
    output reg rx_clk_en,
    output reg[7:0] rx_data,
    output reg busy,
    output reg done
);

reg[7:0] tmp;
reg[3:0] tmpres;

reg [ 1:0] stage;
reg [2:0] curbit;
reg [7:0] rx_rdata;

wire rxd_negedge;
always @(posedge clk or negedge rst_n) begin
    if (!rst_n)
        tmp <= 8'b0;
    else
        tmp <= (tmp << 1) | rxd;
    tmpres <= {tmp[7:6], tmp[1:0]};
end
assign rxd_negedge = ( tmpres === 4'b1100) ? 1'b1 : 1'b0;

always @(posedge clk or negedge rst_n) begin
    if (!rst_n) begin
        busy <= 1'b0;
    end else if (rx_en && rxd_negedge && !busy) begin
        busy <= 1'b1;
    end else if (done)
        busy <= 1'b0;
end

always @(busy) begin
    if (busy)
        rx_clk_en <= 1'b1;
    else
        rx_clk_en <= 1'b0;
end

always @(posedge clk or negedge rst_n) begin
    if (!rst_n) begin
        done <= 1'b0;
        stage <= 2'd0;
        rx_rdata <= 8'd0;
    end else if (busy) begin
        if (rx_clk) begin
            if (stage === 2'd0) begin
                rx_rdata <= 8'd0;
                done <= 1'b0;
                curbit <= 3'd0;
                stage <= 2'b1;
            end else if (stage === 2'b1) begin
                rx_rdata[curbit] <= rxd;
                if (curbit == 3'd7)
                    stage <= 2'd2;
                else
                    curbit <= curbit + 1;
            end else if (stage === 2'd2) begin
                rx_data <= rx_rdata;
                done <= 1'b1;
                stage <= 2'd0;
            end else
                stage <= 2'd0;
        end
    end else begin
        rx_rdata <= 8'd0;
        done <= 1'b0;
        stage <= 2'd0;
    end
end

endmodule

组合
uart.v

module uart(
    input wire clk,
    input wire rst_n,
    input wire rx_en,
    input wire rxd,
    input wire tx_en,
    output wire[7:0] rx_data,
    input wire[7:0] tx_data,
    input  wire[1:0] baud_sel,
    output wire txd,
    output wire rbusy,
    output wire rdone,
    output wire tbusy,
    output wire tdone
);


wire rx_clk,rx_clk_en;
wire tx_clk, tx_clk_en;

uart_clk uclk1(clk,rst_n,tx_clk_en,rx_clk_en,baud_sel,tx_clk,rx_clk);
tx tx1(clk,rst_n,tx_en,tx_clk,tx_data,tx_clk_en,txd,tbusy,tdone);
rx rx1(clk,rst_n,rx_en,rx_clk,rxd,rx_clk_en,rx_data,rbusy,rdone);

endmodule

测试
uart_test.v

module uart_test (
    input wire clk,
    input wire rst_n,
    input wire rxd,
    output wire txd
);
reg rx_en = 1;
reg tx_en;
wire rbusy,tbusy;
wire rdone,tdone;
reg[7:0] tx_data = 0;
wire[7:0] rx_data ;
// reg received = 1'b0;
reg sent = 1'b0;
reg[1:0] baud_sel = 0;


always @(posedge clk or negedge rst_n) begin
    if (!rst_n) begin
        rx_en = 0;
        tx_en = 0;
    end else  if(!tbusy && !rbusy && !sent) begin
        tx_data <= tx_data+1 ;
        tx_en <= 1'b1;
        sent <= 1;
        $display("sent %d",tx_data);
    end else if (rdone) begin
        sent <= 0;
        $display("received %d",rx_data);
    end else begin
        tx_en <= 1'b0;
        rx_en <= 1'b1;
    end
end

uart uart1(clk, rst_n, rx_en,rxd,tx_en,rx_data, tx_data,baud_sel,txd,rbusy,rdone,tbusy,tdone);

endmodule

testbench

module test;

reg rst_n=1 ;

initial begin
    $dumpfile("test.vcd");
    $dumpvars(0, test);

    #10 rst_n = !rst_n;
    #30 rst_n = !rst_n;
    #500000 $finish;
end

reg clk = 0;
always #1 clk = !clk;

wire txd;
wire rxd;
uart_test uart1(clk,rst_n,rxd,txd);
// always @(posedge clk)
//    rxd <= txd;
assign rxd = txd;

endmodule

gtkwave 波形
在这里插入图片描述
方便模拟测试,发送转接收,对比结果。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1341041.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

spring、springmvc、springboot、springcloud简介

spring简介 spring是什么&#xff1f; spring: 春天spring: 轻量级的控制反转和面向切面编程的框架 历史 2002年&#xff0c;首次推出spring雏形&#xff0c;interface 21框架2004年&#xff0c;发布1.0版本Rod Johnson: 创始人&#xff0c;悉尼大学&#xff0c;音乐学博士…

什么是 NLP (自然语言处理)

NLP&#xff08;自然语言处理&#xff09;到底是做什么&#xff1f; NLP 的全称是 Natural Language Processing&#xff0c;翻译成中文称作&#xff1a;自然语言处理。它是计算机和人工智能的一个重要领域。顾名思义&#xff0c;该领域研究如何处理自然语言。 自然语言就是我…

企业如何购买腾讯云服务器?(详细指南)

腾讯云服务器购买流程直接在官方秒杀活动上购买比较划算&#xff0c;在云服务器CVM或轻量应用服务器页面自定义购买价格比较贵&#xff0c;但是自定义购买云服务器CPU内存带宽配置选择范围广&#xff0c;活动上购买只能选择固定的活动机&#xff0c;选择范围窄&#xff0c;但是…

如何本地部署Nextcloud结合cpolar搭建专属私有云盘远程访问(内网穿透)

文章目录 摘要1. 环境搭建2. 测试局域网访问3. 内网穿透3.1 ubuntu本地安装cpolar3.2 创建隧道3.3 测试公网访问 4 配置固定http公网地址4.1 保留一个二级子域名4.1 配置固定二级子域名4.3 测试访问公网固定二级子域名 摘要 Nextcloud,它是ownCloud的一个分支,是一个文件共享服…

前端基本性能指标及lighthouse使用

文章目录 1、基本指标介绍2、Performace分析2.1 performance属性2.2 使用performace计算2.3 Resource Timing API2.4 不重复的耗时时段区分2.5 其他组合分析2.6 JS 总加载耗时2.7 CSS 总加载耗时 3、lighthouse基本使用3.1 使用Chrome插件lighthouse3.2 使用Chrome浏览器开发者…

SQL server使用profiler工具跟踪语句

1.打开SQL server工具&#xff0c;在工具点击SQL server Profiler : 2.流程&#xff1a;暂停——清空———点击接口——启用&#xff1a; 3.找到对应的sql语句&#xff0c;复制到查询界面&#xff0c;开始查询数据&#xff1a;

QML —— ProgressBar示例(附完整源码)

示例 - 效果 实例 - 源码 import QtQuick 2.12 import QtQuick.Window 2.12import QtQuick.Layouts 1.12 import QtQuick.Controls 2.5Window {id: rootIdvisible: truewidth: 640height: 480title: qsTr("Hello World")Column{spacing: 40anchors.centerIn: parent…

UG NX二次开发(C++)-通过两点和高度创建长方体

提示:文章写完后,目录可以自动生成,如何生成可参考右边的帮助文档 文章目录 1、前言2、采用UFun函数来创建长方体3、采用NXOpen方法实现两点和高度创建长方体4、验证1、前言 在UG NX二次开发时,我们通常会采用ufun函数来完成功能的开发,但是有些功能在ufun函数中不能找到…

电子握力器改造

toy_hand_game 介绍 消耗体力玩具&#xff0c;使用握力器(Grip Strengthener)控制舵机旋转。 开始设想是控制丝杆电机滑动&#xff0c;两套设备就可以控制两个丝杆电机进行“模拟拔河”&#xff0c;后续发现硬件设计错误&#xff0c;ULN2003不能控制两相四线电机&#xff0c;…

大数据Doris(四十三):创建物化视图

文章目录 创建物化视图 一、首先你需要有一个Base表

1933年地级市民国铁路开通数据(地级市工具变量)

1933年地级市民国铁路开通数据&#xff08;地级市工具变量&#xff09; 1、时间&#xff1a;1933年 2、指标&#xff1a;是否开通铁路 3、来源&#xff1a;来源于《中国铁道便览》和《中国铁道便览》 4、方法说明&#xff1a;参考田坤&#xff08;2023&#xff09;参考章元…

绝缘电阻测试仪的读数方法有哪些?

绝缘电阻测试仪主要用于测量设备的绝缘电阻&#xff0c;电力工人经常使用它来测试电力设备&#xff0c;从来没有维护过电力设备的正常运行状态。在测试电力设备时&#xff0c;往往需要读取测试结果。绝缘电阻测试仪常用的读数方法有哪些&#xff1f;我一起来盘点一下。 常用的绝…

15 Sequence-Driver-Sequencer communication in UVM

我们分别讨论了sequece_item、sequence、sequencer和driver。在本节中&#xff0c;我们将讨论他们如何相互talk&#xff0c;sequencer如何给driver提供从sequence里的sequence item。在开始阅读本节之前&#xff0c;请确保您了解sequencer和driver中使用的所有方法。&#xff0…

idea Spring Boot项目使用JPA创建与数据库链接

1.pom.xml文件中添加依赖 <dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-data-jpa</artifactId></dependency><dependency><groupId>com.mysql</groupId><artifactId>…

Python+OpenCV 零基础学习笔记(4-5):计算机图形基础+Python相对文件路径+OpenCV图像+OpenCV视频

文章目录 相关链接运行环境前言计算机图形OpenCV简单使用图形读取文件读取可能会出现的问题&#xff1a;路径不对解决方案其它路径问题解决方案 图像显示保存OpenCV视频视频素材如何获取&#xff1f;简单视频读取 相关链接 【2022B站最好的OpenCV课程推荐】OpenCV从入门到实战 …

Jackson通过自定义序列化器给URL拼接访问域名

1、需求 在存储文件访问路径时&#xff0c;一般不会存储域名地址&#xff0c;若文件服务域名和当前应用域名不一致时&#xff0c;就需要在返回数据库的图片路径给前端时&#xff0c;拼接文件服务的域名。 2、处理方式 因为Spring Boot默认使用的是Jackson作为序列化工具&…

redis的搭建及应用(二)-redis的持久化策略

Redis的持久化策略 RDB RDB持久化是指在指定的时间间隔内将redis内存中的数据集快照写入磁盘&#xff0c;实现原理是redis服务在指定的时间间隔内先fork一个子进程&#xff0c;由子进程将数据集写入临时文件&#xff0c;写入成功后&#xff0c;再替换之前的文件&#xff0c;用二…

【力扣】20.有效的括号

家人们&#xff0c;看这排序&#xff0c;一看就很简单&#xff0c;对吧&#xff1f;不对&#xff0c;我觉得还挺不是很容易的&#xff0c;哈哈哈。 题解&#xff1a; 在看题目的时候&#xff0c;我一开始的解题思路就挺复杂的。题目说了”左括号必须以正确的顺序闭合“&#x…

76 Python开发-内外网收集Socket子域名DNS

目录 Python开发相关知识点本篇文章涉及知识点演示案例:IP&Whois&系统指纹获取代码段-外网CDN&子域名&端口扫描&交互代码段-外网IP&计算机名&存活主机&端口扫描代码段-内网Py格式解析环境与可执行程序格式转换-Pyinstaller 涉及资源&#xff1…

我的NPI项目之行业黑话 -- 电子/机构/软件/认证

因为最近的NPI项目&#xff0c;参加了很多项目的会议&#xff0c;有电子/机构/软件/认证相关的各方面的专业词汇就出现了。这里我将之称为黑话&#xff0c;就是对我&#xff08;纯软件) 来说是黑盒的话。这里简单记录并用于理解。 EE有关&#xff1a; Layout&#xff0c;一直…