[X]关闭

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

文档创建者:FPGA课程
浏览次数:144
最后更新:2024-09-20
文档课程分类-安路-DR1
安路-DR1: FPSOC-DR1-FPGA部分 » 2_FPGA实验篇(仅旗舰) » 1-FPGA基础入门实验
软件版本:Anlogic -TD5.9.1-DR1_ES1.1
操作系统:WIN10 64bit
硬件平台:适用安路(Anlogic)FPGA
实验平台:米联客-MLKPAI-SF01-DR1M90M开发板
板卡获取平台:https://milianke.tmall.com/

1概述
本节课继续利用I2C总线控制器实现对RTC时钟芯片,PT7C4337的读写访问,进一步验证我们设计的i2c控制器的可靠性。有了前面的基础,这节课内容学习起来很轻松。本节课主要是基于我们编写的I2C控制器的应用,侧重点是应用,所以不再给出RTL级别的仿真结果,直接通过控制器访问PT7C4337芯片。
在完成本实验前,请确保已经完成前面的实验,包括已经掌握以下能力:
1:完成了TD软件安装
2:完成了modelsim安装以及TD库的编译
3:掌握了TD仿真环境的设置
4:掌握了modesim通过do文件启动仿真
1.1RTC时钟PT7C4337介绍
PT7C4337是低功耗、两线制串行读写接口、日历和时钟数据按BCD码存取的时钟/日历芯片。它提供秒、分、小时、星期、日期、月和年等时钟日历数据。另外它还集成了如下几点功能:
(1)56 字节掉电时电池保持的NV SRAM 数据存储器
(2)可编程的方波信号输出
(3)掉电检测和自动切换电池供电模式

0965e762ae2345cd933a5ba92c42e92d.jpg
引脚说明:
VCC:直流电源,输入范围在1.8~5.5V
GND:接地端
X1/X2:标准的32.768kHz的石英晶振接入端
SCL:串行时钟输入
SDA:串行数据输入/输出
SQW/OUT:方波/中断输出
VBAT:中断输出端
寄存器地址空间:
PT7C4337的寄存器地址空间如下,我们的代码也就是读写以下地址空间。常用的时间寄存器地址为00H~06H,用这七个寄存器来存储秒、分、时、星期、日、月、年,通过写相应的寄存器字段来设置和初始化时间/日历,寄存器的数据格式以BCD码表示(4位二进制数来表示1位十进制数0~9)。
cc65788846e143b1b336c750ad9863b9.jpg
DS1337的读与写
写时序如下:
cd0e51a7c41242578425ad0db72d7235.jpg
写时序很容易理解,和我们前面写EEPROM一样,主器件发起开始条件,先发送器件地址和写命令1101000,等待从器件的应答,再发送寄存器的地址,等待从器件的应答,之后是连续写数据,从器件应答,主器件发送结束信号,代表完成一次写数据过程。
读时序如下:
d137a24e9e524a81b3ba88132da30dd3.jpg
读时序我们采用上图的方式,主器件发起开始条件,先发送器件地址和写命令11010000,等待从器件的应答,再发送寄存器的地址,等待从器件的应答,然后主器件再重新发送开始条件,再发送从器件的地址和读命令11010001,等待从器件的应答,接收从器件发来的数据,主器件应答。这种方式比较方便,大家学习I2C的课程,如果和ZYNQ 部分SDK对比会发现SDK里面操作比较麻烦,而且采取的不是这种方式,这是因为SDK里面的I2C控制不能发送Repeated Start位的原因。
我们这里只针对时、分、秒的读写,而且上电后默认的控制寄存器不需要设置,采用默认参数就可以。
1.2硬件电路分析
因为FPGA一般不需要中断功能,所以中断管脚和方波编程输出引脚没有使用,进行悬空处理。X1、X2引脚接入32.768KHz的晶振构成一个振荡电路,生成芯片所需要的参考时钟信号。IIC总线需要上拉电阻。
990ccab3a8d54b1b85433d710100a0c5.jpg
2用户程序设计
本次实验读写IIC接口的RTC时钟,首先写入一个时间初始值到RTC相应的地址空间,再读取时间值,因为是实时时钟,时钟数据是一直变化的,所以我们需要一直不断的读取时钟的值,并将读取的时间通过串口发送到上位机。实验包含3个模块,I2C MASTER控制器驱动模块、UART发送控制器模块、用户控制模块。以下给出系统框图,关于IIC MASTER控制器驱动、UART发送控制器驱动的详细描述请看前面的实验,我们主要看用户控制模块关于状态机的部分。
306f69d7c8804399bba98539d44c906e.jpg

2.1用户接口时序
先温习下前面课程内容中关于I2C控制器的功能模块可以接口信号:
IO_sda为I2C双向数据总线
O_scl为I2C时钟
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总线忙
04633a18c0da422fb738c281d8136332.jpg
请求一次I2C传输的控制时序如下:
d877a6a08fd84feba178065bb606b3d8.jpg
首先在O_iic_busy=0即I2C总线空闲情况下,设置I_wr_cnt,I_rd_cnt,I_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.2UART发送控制器
将IIC读取到的RTC时间并行数据转成串行数据,并通过UART发送驱动模块,将数据通过uart发送串行总线(O_uart_tx)发回上位机。
7dcfb43881f8433e8733c0bc3c7de35d.jpg
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

e77a5f8add2447ffa333dcd06e4a16ad.jpg
2.4程序源码
  1. `timescale 1ns / 1ns//仿真时间刻度/精度

  2. module rtc_clock_ds1337#
  3. (
  4. parameter SYSCLKHZ   =  25_000_000 //定义系统时钟
  5. )
  6. (
  7. input  wire I_sysclk, //系统时钟输入
  8. output wire O_iic_scl,  //I2C总线,SCL时钟
  9. inout  wire IO_iic_sda, //I2C总线,SDA数据
  10. output wire O_uart_tx   //UART串行发送总线
  11. );

  12. localparam T1000MS_CNT   =  (SYSCLKHZ-1); //定义访问RTC的时间间隔为1000MS
  13. localparam [7:0] RTC_DEV_ADDR =  8'b1101_0000;

  14. reg [8 :0]  rst_cnt       = 9'd0;//上电延迟复位
  15. reg [29:0]  t_cnt = 30'd0;//定时计数器
  16. wire t_en = (t_cnt==T1000MS_CNT);//定时使能  

  17. wire [23:0] wr_data;//写数据信号
  18. wire [23:0] rd_data;//读数据信号
  19. wire        iic_busy;//I2C总线忙
  20. reg  [7 :0] wr_cnt = 8'd0;//写数据计数器
  21. reg  [7 :0] rd_cnt = 8'd0;//读数据计数器
  22. reg         iic_req = 1'b0;//i2c 控制器请求信号
  23. reg  [2 :0] TS_S   = 3'd0;//状态机

  24. reg  [7 :0] rtc_addr;//RTC的寄存器地址
  25. reg         wr_done = 1'b0; //写RTC初值完成信号
  26.          
  27. //初始化时间的BDC码,12:00:00
  28. wire [7 :0] WSecond = {4'd0,4'd0};//妙
  29. wire [7 :0] WMinute = {4'd0,4'd0};//分
  30. wire [7 :0] WHour   = {4'd1,4'd2};//时
  31. reg  [23:0] rtime   = 24'd0; //用于保存读取的时间,格式为BCD码

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

  33. //**********上电延迟复位***************************/
  34. always@(posedge I_sysclk) begin
  35.     if(!rst_cnt[8])
  36.         rst_cnt <= rst_cnt + 1'b1;
  37. end

  38. //**********500ms定时计数器**********************/
  39. always@(posedge I_sysclk) begin
  40.     if(t_cnt == T1000MS_CNT)
  41.         t_cnt <= 0;
  42.     else
  43.         t_cnt <= t_cnt + 1'b1;
  44. end

  45. //读写RTC时钟芯片状态机
  46. always@(posedge I_sysclk) begin
  47.     if(!rst_cnt[8])begin//复位初始化寄存器
  48.         rtc_addr <= 8'd0;
  49.         iic_req  <= 1'b0;
  50.         wr_done  <= 1'b0;
  51.         rd_cnt   <= 8'd0;
  52.         wr_cnt   <= 8'd0;
  53.         TS_S     <= 2'd0;   
  54.     end
  55.     else begin
  56.         case(TS_S)
  57.         0:if(wr_done == 1'b0)begin//上电后,wr_done=0,对RTC时间寄存器初始化,给定初始时间
  58.             wr_done  <= 1'b1;//设置wr_done=1
  59.             rtc_addr <= 8'd0;//设置需要访问的寄存器起始地址
  60.             TS_S     <= 3'd1;//下一个状态
  61.         end
  62.         else begin //已经对RTC芯片初始化完成
  63.             iic_req  <= 1'b0; //重置 iic_req =0
  64.             if(t_en)//每间隔1000ms进行一次读操作
  65.             TS_S     <= 3'd3;//下一个状态,进入读寄时间寄存器状态机
  66.         end
  67.         1:if(!iic_busy)begin//当总线非忙,才可以操作I2C控制器
  68.             iic_req  <= 1'b1;//请求操作I2C控制器
  69.             rd_cnt   <= 8'd0;//由于本操作是写数据,不需要读数据,读数据寄存器设置0
  70.             wr_cnt   <= 8'd5;//需要写入5 BYTES,包括1字节的器件地址,1字节的寄存器起始地址,3字节的BCD时间参数
  71.             TS_S     <= 3'd2;//下一个状态机
  72.         end
  73.         2:if(iic_busy)begin//等待总线忙
  74.             iic_req  <= 1'b0;//重置 iic_req =0
  75.             TS_S     <= 3'd3;//下一个状态机   
  76.         end
  77.         3:if(!iic_busy)begin//该状态读RTC时间寄存器
  78.             iic_req  <= 1'b1;//请求操作I2C控制器
  79.             rtc_addr <= 8'd0;//读RTC寄存器的起始地址
  80.             wr_cnt   <= 8'd2;//读操作需要些1BYTE器件地址,1BYTE 寄存器起始地址
  81.             rd_cnt   <= 8'd3;//读取3个时间寄存器
  82.             TS_S     <= 3'd4;//下一个状态   
  83.         end
  84.         4:if(iic_busy)begin//等待总线空闲
  85.             iic_req  <= 1'b0;//重置 iic_req =0
  86.             TS_S     <= 3'd0;//下一个状态   
  87.         end   
  88.         default: TS_S    <= 3'd0;//default状态回到0
  89.     endcase
  90.    end
  91. end

  92. //***********保存从RTC读取到的时间寄存器,时间为BCD格式***********//
  93. always@(posedge I_sysclk) begin
  94.     if(!rst_cnt[8])
  95.         rtime <=0;
  96.    else if(TS_S == 3)
  97.         rtime[23: 0] <= rd_data;//读取的时间包括 时:分:秒,BCD格式
  98. end

  99. //例化I2C控制模块
  100. uii2c#
  101. (
  102. .WMEN_LEN(5),//最大支持一次写入4BYTE(包含器件地址)
  103. .RMEN_LEN(3),//最大支持一次读出3BYTE
  104. .CLK_DIV(SYSCLKHZ/25000)//100KHZ I2C总线时钟
  105. )
  106. uii2c_inst
  107. (
  108. .I_clk(I_sysclk),//系统时钟
  109. .I_rstn(rst_cnt[8]),//系统复位
  110. .O_iic_scl(O_iic_scl),//I2C SCL总线时钟
  111. .IO_iic_sda(IO_iic_sda),//I2C SDA数据总线
  112. .I_wr_data({wr_data,rtc_addr,RTC_DEV_ADDR}),//写数据寄存器
  113. .I_wr_cnt(wr_cnt),//需要写的数据BYTES
  114. .O_rd_data(rd_data), //读数据寄存器
  115. .I_rd_cnt(rd_cnt),//需要读的数据BYTES
  116. .I_iic_req(iic_req),//I2C控制器请求
  117. .I_iic_mode(1'b1),//读模式
  118. .O_iic_busy(iic_busy)//I2C控制器忙
  119. //.iic_bus_error(iic_bus_error),//总线错误信号标志
  120. //.IO_iic_sda_dg(IO_iic_sda_dg)//debug IO_iic_sda
  121. );

  122. //以下完成BCD码赚ASCII码,这样通过串口打印可以方便观察
  123. function signed[7:0] ascii ;   //定义ascii码转换函数,只需要转换BCD数据
  124.    
  125. input[7:0] bcd; //输入参数  

  126. begin                                                   
  127.     case(bcd)
  128.     0 :     ascii   =   {8'h30};//ascii 码0  
  129.     1 :     ascii   =   {8'h31};//ascii 码1      
  130.     2 :     ascii   =   {8'h32};//ascii 码2  
  131.     3 :     ascii   =   {8'h33};//ascii 码3
  132.     4 :     ascii   =   {8'h34};//ascii 码4  
  133.     5 :     ascii   =   {8'h35};//ascii 码5         
  134.     6 :     ascii   =   {8'h36};//ascii 码6  
  135.     7 :     ascii   =   {8'h37};//ascii 码7  
  136.     8 :     ascii   =   {8'h38};//ascii 码8  
  137.     9 :     ascii   =   {8'h39};//ascii 码9
  138.     default:ascii   =   {8'h00};   
  139.     endcase                                
  140. end  
  141.                                                    
  142. endfunction   

  143. //例化UART发送模块
  144. uart_tx_block #
  145. (
  146. .TX_BYTES(10),//设置需要发送的字节数
  147. .BAUD_DIV(SYSCLKHZ/115200 -1) //设置串口波特率
  148. )
  149. u_uart_tx_block
  150. (
  151. .I_sysclk(I_sysclk),//系统时钟输入
  152. .O_uart_tx(O_uart_tx),//UART 串行总线数据发送
  153. //高位,8'h0a,8'h0d,为回车+换行控制字符
  154. .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])}),
  155. .I_uart_tx_buf_en(t_en)//t_en也是发送使能
  156. );

  157. endmodule
复制代码
3FPGA工程
fpga工程的创建过程不再重复
bcd83666242f4d1d958aca1731928b41.jpg
米联客的代码管理规范,在对应的FPGA工程路径下创建uisrc路径,并且创建以下文件夹
01_rtl:放用户编写的rtl代码
02_sim:仿真文件或者工程
03_ip:放使用到的ip文件
04_pin:放fpga的pin脚约束文件或者时序约束文件
05_boot:放编译好的bit或者bin文件(一般为空)
06_doc:放本一些相关文档(一般为空)
506fd76fe6fb4c0abe647a0cdb62c5aa.jpg
4下载演示4.1硬件连接
(该教程为通用型教程,教程中仅展示一款示例开发板的连接方式,具体连接方式以所购买的开发板型号以及结合配套代码管脚约束为准。)
请确保下载器和开发板已经正确连接,并且开发板已经上电(注意JTAG端子不支持热插拔,而USB接口支持,所以在不通电的情况下接通好JTAG后,再插入USB到电脑,之后再上电,以免造成JTAG IO损坏)
a03f66f270dd4360adf4ee73a172fd33.jpg
4.2运行结果
cdd02571f2d24166907d8e3c1318efb0.jpg
您需要登录后才可以回帖 登录 | 立即注册

本版积分规则