[米联客-安路飞龙DR1-FPSOC] FPGA基础篇连载-20 读写I2C接口的RTC时钟芯片

news/2024/11/15 13:48:20/文章来源:https://www.cnblogs.com/milianke/p/18330489

软件版本:Anlogic -TD5.9.1-DR1_ES1.1

操作系统:WIN10 64bit

硬件平台:适用安路(Anlogic)FPGA

实验平台:米联客-MLK-L1-CZ06-DR1M90G开发板

板卡获取平台:https://milianke.tmall.com/

登录"米联客"FPGA社区 http://www.uisrc.com 视频课程、答疑解惑!

 

1概述

    本节课继续利用I2C总线控制器实现对RTC时钟芯片,DS1337的读写访问,进一步验证我们设计的i2c控制器的可靠性。有了前面的基础,这节课内容学习起来很轻松。本节课主要是基于我们编写的I2C控制器的应用,侧重点是应用,所以不再给出RTL级别的仿真结果,直接通过控制器访问DS1337芯片。

在完成本实验前,请确保已经完成前面的实验,包括已经掌握以下能力:

1:完成了TD软件安装

2:完成了modelsim安装以及TD库的编译

3:掌握了TD仿真环境的设置

4:掌握了modesim通过do文件启动仿真

1.1 RTC时钟DS1337介绍

    DS1337是低功耗、两线制串行读写接口、日历和时钟数据按BCD码存取的时钟/日历芯片。它提供秒、分、小时、星期、日期、月和年等时钟日历数据。另外它还集成了如下几点功能:

156字节掉电时电池保持的SRAM 数据存储器

2)可编程的方波信号输出

3)掉电检测和自动切换电池供电模式

引脚说明:

VCC:直流电源,输入范围在1.8~5.5V

GND:接地端

X1/X2:标准的32.768kHz的石英晶振接入端

SCL:串行时钟输入

SDA:串行数据输入/输出

SQW/OUT:方波/中断输出

VBAT:中断输出端

寄存器地址空间:

DS1337的寄存器地址空间如下,我们的代码也就是读写以下地址空间。常用的时间寄存器地址为00H~06H,用这七个寄存器来存储秒、分、时、星期、日、月、年,通过写相应的寄存器字段来设置和初始化时间/日历,寄存器的数据格式以BCD码表示(4位二进制数来表示1位十进制数0~9)。

 

地址空间中详细的参数定义如下表

00H/01H分寄存器:计数范围为00~59BIT0~BIT3表示个位。BIT4~BIT6表示十位。

02H小时寄存器:BIT6被定义为12小时模式("1")或24小时模式("0")选择位。在12小时模式时,计数范围为1~12BIT5AM("0")或PM("1")标志位;当在24小时模式时,计数范围为00~23,需要6位才能存下0~23的数据,BIT5是第二个十位(表示2023小时)

03H星期寄存器:计数范围为1~7BIT0~BIT3表示个位。

04H日寄存器:计数范围为01~31BIT0~BIT3表示个位,BIT4~BIT5表示十位。

05H月寄存器:BIT7是世纪位,当年寄存器从099溢出时,该位发生变化。

06H年寄存器:计数范围为00~99BIT0~BIT3表示个位,BIT4~BIT7表示十位。

注意:07H~0FH为控制寄存器和状态寄存器,用于控制实时时钟、闹钟和方波的输出,本实验并未使用,详细介绍自行查阅芯片手册。

 

DS1337的读与写:

写时序如下:

写时序很容易理解,和我们前面写EEPROM一样,主器件发起开始条件,先发送器件地址和写命令1101000,等待从器件的应答,再发送寄存器的地址,等待从器件的应答,之后是连续写数据,从器件应答,主器件发送结束信号,代表完成一次写数据过程。

读时序如下:

读时序我们采用上图的方式,主器件发起开始条件,先发送器件地址和写命令11010000,等待从器件的应答,再发送寄存器的地址,等待从器件的应答,然后主器件再重新发送开始条件,再发送从器件的地址和读命令11010001,等待从器件的应答,接收从器件发来的数据,主器件应答。这种方式比较方便,大家学习I2C的课程,如果和ZYNQ 部分SDK对比会发现SDK里面操作比较麻烦,而且采取的不是这种方式,这是因为SDK里面的I2C控制不能发送Repeated Start位的原因。

我们这里只针对时、分、秒的读写,而且上电后默认的控制寄存器不需要设置,采用默认参数就可以。

1.2 DS1337时序

根据手册我们可以知道I2C时钟频率、建立时间、保持时间、两次读写之间的时间间隔有没有限制等。由图可知DS1337I2C支持快速和标准模式,串行时钟SCL频率可以位于0~400KHz。而建立时间Tsu和保持时间Thd在快速模式下都小于1usI2C接口在发送一个STOP条件后到发送下一个START条件之间的最小间隔时间(Bus Free Time Between a STOP and START Condition)在快速模式(Fast mode)下,不小于1.3微秒、,而在标准模式(Standard mode)下,不小于4.7微秒,这段时间内总线上必须保持空闲。

 

1.3硬件电路分析

因为FPGA一般不需要中断功能,所以中断管脚和方波编程输出引脚没有使用,进行悬空处理。X1X2引脚接入32.768KHz的晶振构成一个振荡电路,生成芯片所需要的参考时钟信号。IIC总线需要上拉电阻。

2用户程序设计

本次实验读写IIC接口的RTC时钟,首先写入一个时间初始值到RTC相应的地址空间,再读取时间值,因为是实时时钟,时钟数据是一直变化的,所以我们需要一直不断的读取时钟的值,并将读取的时间通过串口发送到上位机。实验包含3个模块,I2C MASTER控制器驱动模块、UART发送控制器模块、用户控制模块。以下给出系统框图,关于IIC MASTER控制器驱动、UART发送控制器驱动的详细描述请看前面的实验,我们主要看用户控制模块关于状态机的部分。

 

2.1用户接口时序

先温习下前面课程内容中关于I2C控制器的功能模块可以接口信号:

IO_sdaI2C双向数据总线

O_sclI2C时钟

I_wr_cnt写数据字节长度,包含了器件地址,发送I_iic_req前,预设该值

I_rd_cnt读数据字节长度,仅包含读回有效部分,发送I_iic_req前,预设该值

I_wr_data写入的数据

O_rd_data读出的数据,如果是读请求,当O_iic_busy从高变低代表数据读回有效

I_iic_req I2C操作请求,根据I_rd_cnt是否大于0决定是否有读请求

I_iic_mode是否支持随机读写,发送I_iic_req前,预设该值

O_iic_busy总线忙

请求一次I2C传输的控制时序如下:

首先在O_iic_busy=0I2C总线空闲情况下,设置I_wr_cntI_rd_cntI_wr_data,并且设置I_iic_req=1,启动I2C传输。当O_iic_busy=1说明I2C控制器开始传输,这时候可以设置I_iic_req=0,结束本次请求,并且等待O_iic_busy=0,当O_iic_busy=0代表本次传输结束.如果发送的是读请求(I_rd_cnt>0),则此时O_rd_data有效可以读走数据。

2.2 UART发送控制器

IIC读取到的RTC时间并行数据转成串行数据,并通过UART发送驱动模块,将数据通过uart发送串行总线(O_uart_tx)发回上位机。

2.3 用户控制状态机

首先进行通过复位进行数据初始化。

TS_S=0:板子刚上电时,首先初始化RTC寄存器的时间,设置需要访问的寄存器起始地址,并进入到下一个状态;如果已经完成对RTC芯片初始化,等待1000ms定时使能信号拉高,再次读取RTC寄存器的数据。

TS_S=1:当总线非忙,设置iic_req=1,请求操作I2C控制器,并设置需要写入的字节数(此时不需要读操作)。

TS_S=2:总线忙,重置iic_req =0,此时为写数据过程。

TS_S=3:等待总线非忙,代表前面写数据已完成,设置iic_req=1请求操作I2C控制。读操作前需要先写1BYTE器件地址,1BYTE 寄存器起始地址,再读取3个时间寄存器。

TS_S=4:等待总线非忙,表示一次读RTC寄存器操作完成。回到状态0

 

2.4程序源码

`timescale 1ns / 1ns//仿真时间刻度/精度

 

modulertc_clock_ds1337#

(

parameter SYSCLKHZ   =  25_000_000//定义系统时钟

)

(

input  wire I_sysclk, //系统时钟输入

outputwire O_iic_scl,  //I2C总线,SCL时钟

inout  wire IO_iic_sda, //I2C总线,SDA数据

outputwire O_uart_tx   //UART串行发送总线

);

 

localparam T1000MS_CNT   =  (SYSCLKHZ-1); //定义访问RTC的时间间隔为1000MS

localparam [7:0] RTC_DEV_ADDR =  8'b1101_0000;

 

reg [8 :0]  rst_cnt       = 9'd0;//上电延迟复位

reg [29:0]  t_cnt = 30'd0;//定时计数器

wire t_en = (t_cnt==T1000MS_CNT);//定时使能  

 

wire [23:0] wr_data;//写数据信号

wire [23:0] rd_data;//读数据信号

wire        iic_busy;//I2C总线忙

reg  [7 :0] wr_cnt = 8'd0;//写数据计数器

reg  [7 :0] rd_cnt = 8'd0;//读数据计数器

reg         iic_req = 1'b0;//i2c 控制器请求信号

reg  [2 :0] TS_S   = 3'd0;//状态机

 

reg  [7 :0] rtc_addr;//RTC的寄存器地址

reg         wr_done = 1'b0; //RTC初值完成信号

           

//初始化时间的BDC码,12:00:00

wire [7 :0] WSecond = {4'd0,4'd0};//

wire [7 :0] WMinute = {4'd0,4'd0};//

wire [7 :0] WHour   = {4'd1,4'd2};//

reg  [23:0] rtime   = 24'd0; //用于保存读取的时间,格式为BCD

 

assign wr_data   = {WHour,WMinute,WSecond};//写数据初值

 

//**********上电延迟复位***************************/

always@(posedge I_sysclk) begin

    if(!rst_cnt[8])

        rst_cnt <= rst_cnt + 1'b1;

end

 

//**********500ms定时计数器**********************/

always@(posedge I_sysclk) begin

    if(t_cnt == T1000MS_CNT)

        t_cnt <= 0;

    else

        t_cnt <= t_cnt + 1'b1;

end

 

//读写RTC时钟芯片状态机

always@(posedge I_sysclk) begin

    if(!rst_cnt[8])begin//复位初始化寄存器

        rtc_addr <= 8'd0;

        iic_req  <= 1'b0;

        wr_done  <= 1'b0;

        rd_cnt   <= 8'd0;

        wr_cnt   <= 8'd0;

        TS_S     <= 2'd0;    

    end

    elsebegin

        case(TS_S)

        0:if(wr_done == 1'b0)begin//上电后,wr_done=0,对RTC时间寄存器初始化,给定初始时间

            wr_done  <= 1'b1;//设置wr_done=1

            rtc_addr <= 8'd0;//设置需要访问的寄存器起始地址

            TS_S     <= 3'd1;//下一个状态

        end

        elsebegin//已经对RTC芯片初始化完成

            iic_req  <= 1'b0; //重置 iic_req =0

            if(t_en)//每间隔1000ms进行一次读操作

            TS_S     <= 3'd3;//下一个状态,进入读寄时间寄存器状态机

        end

        1:if(!iic_busy)begin//当总线非忙,才可以操作I2C控制器

            iic_req  <= 1'b1;//请求操作I2C控制器

            rd_cnt   <= 8'd0;//由于本操作是写数据,不需要读数据,读数据寄存器设置0

            wr_cnt   <= 8'd5;//需要写入5 BYTES,包括1字节的器件地址,1字节的寄存器起始地址,3字节的BCD时间参数

            TS_S     <= 3'd2;//下一个状态机

        end

        2:if(iic_busy)begin//等待总线忙

            iic_req  <= 1'b0;//重置 iic_req =0

            TS_S     <= 3'd3;//下一个状态机  

        end

        3:if(!iic_busy)begin//该状态读RTC时间寄存器

            iic_req  <= 1'b1;//请求操作I2C控制器

            rtc_addr <= 8'd0;//RTC寄存器的起始地址

            wr_cnt   <= 8'd2;//读操作需要些1BYTE器件地址,1BYTE 寄存器起始地址

            rd_cnt   <= 8'd3;//读取3个时间寄存器

            TS_S     <= 3'd4;//下一个状态    

        end

        4:if(iic_busy)begin//等待总线空闲

            iic_req  <= 1'b0;//重置 iic_req =0

            TS_S     <= 3'd0;//下一个状态    

        end  

        default: TS_S    <= 3'd0;//default状态回到0

    endcase

   end

end

 

//***********保存从RTC读取到的时间寄存器,时间为BCD格式***********//

always@(posedge I_sysclk) begin

    if(!rst_cnt[8])

        rtime <=0;

   elseif(TS_S == 3)

        rtime[23: 0] <= rd_data;//读取的时间包括 时:分:秒,BCD格式

end

 

//例化I2C控制模块

uii2c#

(

.WMEN_LEN(5),//最大支持一次写入4BYTE(包含器件地址)

.RMEN_LEN(3),//最大支持一次读出3BYTE

.CLK_DIV(SYSCLKHZ/100000)//100KHZ I2C总线时钟

)

uii2c_inst

(

.I_clk(I_sysclk),//系统时钟

.I_rstn(rst_cnt[8]),//系统复位

.O_iic_scl(O_iic_scl),//I2C SCL总线时钟

.IO_iic_sda(IO_iic_sda),//I2C SDA数据总线

.I_wr_data({wr_data,rtc_addr,RTC_DEV_ADDR}),//写数据寄存器

.I_wr_cnt(wr_cnt),//需要写的数据BYTES

.O_rd_data(rd_data), //读数据寄存器

.I_rd_cnt(rd_cnt),//需要读的数据BYTES

.I_iic_req(iic_req),//I2C控制器请求

.I_iic_mode(1'b1),//读模式

.O_iic_busy(iic_busy)//I2C控制器忙

//.iic_bus_error(iic_bus_error),//总线错误信号标志

//.IO_iic_sda_dg(IO_iic_sda_dg)//debug IO_iic_sda

);

 

//以下完成BCD码赚ASCII码,这样通过串口打印可以方便观察

functionsigned[7:0] ascii ;   //定义ascii码转换函数,只需要转换BCD数据

     

input[7:0] bcd; //输入参数  

 

begin                                                    

    case(bcd)

    0 :     ascii   =   {8'h30};//ascii 0  

    1 :     ascii   =   {8'h31};//ascii 1      

    2 :     ascii   =   {8'h32};//ascii 2  

    3 :     ascii   =   {8'h33};//ascii 3

    4 :     ascii   =   {8'h34};//ascii 4  

    5 :     ascii   =   {8'h35};//ascii 5          

    6 :     ascii   =   {8'h36};//ascii 6  

    7 :     ascii   =   {8'h37};//ascii 7  

    8 :     ascii   =   {8'h38};//ascii 8  

    9 :     ascii   =   {8'h39};//ascii 9

    default:ascii   =   {8'h00};    

    endcase                                

end  

                                                     

endfunction  

 

//例化UART发送模块

uart_tx_block #

(

.TX_BYTES(10),//设置需要发送的字节数

.BAUD_DIV(SYSCLKHZ/115200 -1) //设置串口波特率

)

u_uart_tx_block

(

.I_sysclk(I_sysclk),//系统时钟输入

.O_uart_tx(O_uart_tx),//UART 串行总线数据发送

//高位,8'h0a,8'h0d,为回车+换行控制字符

.I_uart_tx_buf({8'h0a,8'h0d,ascii(rtime[3:0]),ascii(rtime[7:4]),8'h2d,ascii(rtime[11:8]),ascii(rtime[15:12]),8'h2d,ascii(rtime[19:16]),ascii(rtime[23:20])}),

.I_uart_tx_buf_en(t_en)//t_en也是发送使能

);

 

endmodule

3 FPGA工程

fpga工程的创建过程不再重复

米联客的代码管理规范,在对应的FPGA工程路径下创建uisrc路径,并且创建以下文件夹

01_rtl:放用户编写的rtl代码

02_sim:仿真文件或者工程

03_ip:放使用到的ip文件

04_pin:fpgapin脚约束文件或者时序约束文件

05_boot:放编译好的bit或者bin文件(一般为空)

06_doc:放本一些相关文档(一般为空)

4下载演示

4.1硬件连接

(该教程为通用型教程,教程中仅展示一款示例开发板的连接方式,具体连接方式以所购买的开发板型号以及结合配套代码管脚约束为准。)

请确保下载器和开发板已经正确连接,并且开发板已经上电(注意JTAG端子不支持热插拔,而USB接口支持,所以在不通电的情况下接通好JTAG后,再插入USB到电脑,之后再上电,以免造成JTAG IO损坏)

 

4.2运行结果

在配套demo路径下我们也提供了可以用串口打印的demo

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

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

【YashanDB知识库】如何远程连接、使用YashanDB?

问题现象 在各个项目实施中,我们经常遇到客户、开发人员需要连接和使用YashanDB但不知如何操作的问题,本文旨在介绍远程连接、使用YashanDB的几种方式。 问题的风险及影响 无风险 问题影响的版本 历史版本~23.2 问题发生原因 无 解决方法及规避方式 不需要规避 问题分析和处理…

GIS场景零代码拖拽式编辑,支持TMS/WMS/WMTS等多种GIS协议

在三维GIS领域,编辑场景和处理影像数据通常是一个复杂且费时的过程,但现在有了山海鲸可视化,这一切都变得简单有趣。这款免费可视化工具为您提供了零代码拖拽式编辑的体验,让您无需编程知识就能轻松创建和优化GIS场景。通过直观的界面,您只需动动鼠标就能完成从场景编辑到…

【攻防技术系列+代理转发】ICMPSH 工具

虚拟机环境搭建: 【Kali】,192.168.10.131 【window7】,192.168.10.1工具: ICMPSH(基于网络层) Wireshark实验开始前,确保两台主机可以ping通。如果遇到环境ping不通的情况,可以借鉴以下解决方案。 重启网卡:有来有回的过程,数据长度为74。 【kali】: sysctl -w net…

人大金仓踩坑指南

现在越来越多的项目要求-切换国产数据库系统,或达梦(性能高),或人大金仓(扩展性高,更稳定),跨平台如果上来就干,坑还是不少的,本人有幸踩个遍.1. 软件有效期,人大金仓可不便宜,不少政府采购网上都是标价10万+, 测试时候用专业版玩一玩就好到期后,可以直接下载其他版…

linux shell read 按列读取txt文本

前言全局说明一、说明 通常情况下,如果文本里有多列数据,会先读入,然后用 grep 和 awk 先拆分成行,在拆分成列。这样费时费力,遇到特殊字符行,还不好处理。 在解决别的问题时候,无意发现 read 有直接按列读取的功能。二、文件 2.1 存放两列数据的文件 文件名:list.txt…

什么是云计算?

云计算是一种服务的模式(商业模式):将信息类资源以服务的方式提供给用户使用,用户可以便捷的、按需计费的、弹性的从云端获取到信息技术的服务。 云计算技术栈层级:CPU CPU的组成: 1、运算器(算术逻辑单元) 2、控制器() 3、存储器(命令和需要运算的数据) 1、高速缓…

Linux shell mktemp -d命令生成临时文件

前言全局说明一、说明二、mktemp 命令 2.1 创建临时文件 mktemp 2.1 创建临时目录 mktemp -d三、命令行示例 mktemp ll /tmp/tmp.fvi5gFbDgr四、sh 脚本使用 4.1 创建 tmpfile=$(mktemp)4.2 删除 rm "$tmpfile"免责声明:本号所涉及内容仅供安全研究与教学使用,如出…

Linux shell mktemp命令生成临时文件

前言全局说明一、说明二、mktemp 命令 mktemp三、命令行示例 mktemp ll /tmp/tmp.fvi5gFbDgr四、sh 脚本使用 4.1 创建 tmpfile=$(mktemp)4.2 删除 rm "$tmpfile"免责声明:本号所涉及内容仅供安全研究与教学使用,如出现其他风险,后果自负。参考、来源: ChatGPT 作…

「代码随想录算法训练营」第二十三天 | 贪心算法 part1

455. 分发饼干题目链接:https://leetcode.cn/problems/assign-cookies/ 题目难度:简单 文章讲解:https://programmercarl.com/0455.分发饼干.html 视频讲解:https://www.bilibili.com/video/BV1MM411b7cq 题目状态:初次有贪心算法的总体概念,有点懵思路: 先将饼干尺寸大…

人工智能|利用人工智能自动找bug

简介 在程序员编程的过程中,产生Bug是一件稀松平常的事情,以前在编码的过程中提前找出Bug,需要通过单元测试、CodeReview等各种方式。 当今,人工智能技术的发展给软件开发和测试带来了许多机会。利用人工智能技术,可以开发出自动化的 bug 检测工具,从而提高软件质量和可靠…

Pentester Academy -Windows API Exploitation Recipes: Processes, Tokens and Memory RW 2017版本

早年为Pentester Academy(https://www.pentesteracademy.com/) ,如今为INE (https://ine.com/)002 安装VS社区版 https://visualstudio.microsoft.com/zh-hans/003 process listing api 正在运行的是什么:服务,AV,HIDS/IPS等 其他attack开始的点:进程注入,内存dump/修改,…

基于EasyTcp4Net开发一个功能较为完善的去持久化聊天软件

之前自己写了一篇介绍TCP的一些常用的功能介绍和特征,并且用代码做了示例,最终开发了一个EasyTcp4Net的TCP工具库,其最大的特色就是使用了微软提供的高性能库中的一些数据结构来处理TCP数据。 最近辞职待业在家,也没啥事做,就利用自己写的TCP通讯库基础上开发了一个示例的…