基于STM8的IIC协议--实例篇--时钟模块(DS3231)读取

1. 综述

  由上篇博客可知道IIC协议如何用代码实现,本篇博客就不涉及协议内容,只讲解如何使用。

  本次的实验传感为:DS3231(时钟模块),对于时钟模块的具体信息我也就不多介绍大家可以自行度娘,具体功能无非就是在单片机中起到一个获取时间的作用。然后该模块是可以由IIC协议去驱动的,再加上所要的操作也是比较简单,适合部分刚接触IIC协议而找不但传感练手的一个模块。

2. 明确任务顺序

  个人习惯,在每驱动一个新传感的时候,我会将我要完成的传感分为几个任务点。接下来我就展示以下我在写DS3231模块时的任务栏。

一.硬件部分
1.完成排针焊接
2.完成传感引角和单片机引角的接线
二.软件部分
1. 找到该传感的官方手册并认真阅读
2. 找到并定义该传感的地址(7位地址、8位地址)3. 找到并定义该传感器的寄存器地址
4. 完成读写函数的代码
5. 测试传感是否有响应6. 查看手册,了解传感的工作模式7. 完成传感参数初始化
8. 传感数据的获取

  根据上面这些小任务一一解决,这样一来,大多数传感就能成功驱动了,这里在给一个建议,驱动传感器的代码最好不要一次性全部写完再进行测试,这样成功率不高且大大增加你找问题的难度,将每个功能函数测试完再进行下一个功能函数的编写,会大大加大你的效率。

3. 具体任务实现步骤

  3.1. 硬件部分

  在STM8S103芯片中,我所定义的IIC协议中的SDA引角为PD3,SCL引角为PD4,该时钟模块所需提供的电源为3.3V。所以接线图如下。

  

  3.2. 软件部分

 (1)官方手册

  我这里提供的是全英版的官方手册,多看点英文对你有好处的。

链接:https://pan.baidu.com/s/1Oo6o68SoVr7gt8tPZIoQxw
提取码:uzih 

  (2)器件地址

  在说明手册中,我们可以在第16页的右下角找到下面这句话:

  The slave address byte contains the 7-bit DS3231 address, which is 1101000, followed by the direction bit (R/W), which is 0 for a write.

  这句话的大意为,从机的7位地址为1101000,后跟一位读写位,读写位为0时是写位。即7位地址的为0x68,8位地址为0xD0。

  (3)寄存器地址

  

  DS3231的寄存器地址都在这了,大家可根据所需要用的功能定义相应的寄存器。

  以下是我在头文件所定义的格式,因为我只需要获取时间,不需要进行其他操作,所以部分寄存器没有使用。

1 #define DS3231_Address           0xD0                           //设备地址
2 #define Seconds_Register         0x00                           //秒钟寄存器
3 #define Minutes_Register         0x01                           //分钟寄存器
4 #define Hour_Register            0x02                           //小时寄存器
5 #define Day_Register             0x03                           //星期寄存器
6 #define Date_Register            0x04                           //日期寄存器
7 #define Month_Register           0x05                           //月份寄存器(第七位是世纪位)
8 #define Year_Register            0x06                           //年份寄存器

4. 例程

4.1 编译环境

  我的编译环境是IAR,这款软件是现在STM8的主流平台,比较推荐。不过我打算等到STCubeMX更新出比较方便的版本后再去使用Keil5,因为我在用STM32的时候就是利用Keil5,的确很方便,你们也可以学着用一下。

4.2 主芯片

  我的主芯片是STM8S系列中的103,其中STM8S的003、005、和103、105,配置一样(外设和CPU频率,FLASH),在代码相同的情况下均可进行烧写。

4.3 库文件的添加

  我们的工程可以在IAR中的官方例程中复制,操作过程:打开STM8S_StdPeriph_Lib(这是一个官方的库文件,下载IAR STM8包的时候就携带,里面有库文件和相对应的例程),将Libraries文件复制到你工程所在的文件下,并将有关于ADC的库文件添加到你的工程列表当中。添加完成后,就可以开始编写代码了(如果你将全部的库文件都添加进来的话,编译程序后库文件还有红点报错的话,这是因为你选的芯片上没有该功能,你需要将其删掉才能不报错。)如图。

4.4 代码

4.4.1 SDA、SCL引角的定义

  我这里将SDA、SCL都设置为了推挽输出,具体为什么可与参考上一片IIC协议讲解。

1   //IIC引脚
2   GPIO_Init(IIC_SCL_GPIO_Port, IIC_SCL_Pin, GPIO_MODE_OUT_PP_HIGH_FAST);
3   GPIO_Init(IIC_SDA_GPIO_Port, IIC_SDA_Pin, GPIO_MODE_OUT_PP_HIGH_FAST);

4.4.2 DS3231句柄定义

  为了能够方便数据的管理,所以我定义了DS3231的一个句柄。

 1 typedef struct DS3231
 2 {
 3   uint8_t uSeconds;                     //秒
 4   uint8_t uMinutes;                     //分钟
 5   uint8_t uHour;                        //小时
 6   uint8_t uDay;                         //星期
 7   uint8_t uData;                        //日期
 8   uint8_t uMouth;                       //月份
 9   uint8_t uyear;                        //年份
10   uint8_t uTime[3];                     //将秒、分钟、小时、日期包括在内
11
12
13 }DS3231_HandleTypeDef; 

4.4.3 BCD格式和B格式转换

  在手册中可以看到,写入到DS3231中的格式是BCD模式,而读取到的却是B格式,所以我们需要两个函数将其转换,也方便我们察看数据的结果。

 1 uint8_t Byte_Transform_BCD(uint8_t uData)
 2 {
 3   uint8_t i, j, uBCD_Code;
 4
 5   i = uData / 10;
 6   j = uData % 10;
 7   uBCD_Code = j + ( i << 4 );
 8
 9   return uBCD_Code;
10
11 }
12
13 uint8_t BCD_Transform_Byte(uint8_t uData)
14 {
15     uint8_t uByte_Code;
16
17     uByte_Code  = (uData & 0x0f);
18     uData     >>= 4;
19     uData      &= 0x0f;
20     uData      *= 10;
21     uByte_Code += uData;
22
23     return uByte_Code;
24
25 }

4.4.4读写函数编写

  读写函数我们如果不去看手册说明的话是无法编写代码的,所以我们需要去看手册,根据手册中的IIC命令顺序进行编写代码。

    这幅图为 IIC数据传输概述。

    写函数流程图:

    

  这个流程图是在DS3231中的官方图,根据图中的命令可以写出代码,其中,代码中出现IIC_HandleTypedef * iicHandle等参数,是IIC的句柄,具体可看上一篇博客,下面所出现的也是一样的意思。

 1 uint8_t vSen_DS3231_Write_Bytes(IIC_HandleTypedef * iicHandle, uint8_t Register_Address, uint8_t Data_Byte)
 2 {
 3
 4   vIIC_Start_Signal(iicHandle);                                                 //1.  IIC_Start                 ;  起始信号
 5   vIIC_SendByte(iicHandle, Slave_Address);                                      //2.  IIC_Send Device Address(W);  发送(设备地址)告诉总线即将操作的设备
 6
 7   if(!bIIC_ReadACK(iicHandle))                                                  //3.  IIC_ReadAck               ;  等待响应
 8   {
 9     vIIC_Stop_Signal(iicHandle);
10     return FALSE;
11   }
12
13   vIIC_SendByte(iicHandle, Register_Address);                                   //4.  IIC_Send Register Address ; 发送(寄存器)   告诉设备我们即将操作的寄存器
14   bIIC_ReadACK(iicHandle);                                                      //5.  IIC_ReadAck               ; 等待响应
15   vIIC_SendByte(iicHandle, Data_Byte);                                          //6.  IIC_Send the data to Reg  ; 发送(数据)     写入数据到指定设备的寄存器中
16   bIIC_ReadACK(iicHandle);                                                      //7.  IIC_ReadAck               ; 等待响应
17   vIIC_Stop_Signal(iicHandle);                                                  //8.  IIC_Stop                  ; 结束信号
18
19
20   return TRUE;
21
22 }

  读函数流程图:

 1 uint8_t vSen_DS3231_Read_Bytes(IIC_HandleTypedef * iicHandle, uint8_t Register_Address)
 2 {
 3   uint8_t uRev_Register_Data = 0x00;
 4
 5   vIIC_Start_Signal(iicHandle);                                                 //1.  IIC_Start                 ;  起始信号
 6   vIIC_SendByte(iicHandle, Slave_Address);                                      //2.  IIC_Send Device Address(W);  发送(设备地址)告诉总线即将操作的设备
 7
 8   if(!bIIC_ReadACK(iicHandle))                                                  //3.  IIC_ReadAck               ;  等待响应
 9   {
10     vIIC_Stop_Signal(iicHandle);
11     return FALSE;
12   }
13
14   vIIC_SendByte(iicHandle, Register_Address);                                   //4.  IIC_Send Register Address ; 发送(寄存器)   告诉设备我们即将操作的寄存器
15   vIIC_Ack(iicHandle);                                                          //5.  IIC_Ack                   ; 主动响应
16   vIIC_Start_Signal(iicHandle);                                                 //6.  IIC_Start                 ; 起始信号17   vIIC_SendByte(iicHandle, Slave_Address+1);                                    //7.  IIC_Send Device Address(R); 发送(设备地址)告诉总线即将操作的设备
18   vIIC_Ack(iicHandle);                                                          //8.  IIC_Ack                   ; 主动响应
19   uRev_Register_Data = uIIC_RecvByte(iicHandle);                                //9.  IIC_ReadByte              ; 读取寄存器中的数据
20   vIIC_NAck(iicHandle);                                                         //10. IIC_Nack                  ; 主动不应答
21   vIIC_Stop_Signal(iicHandle);                                                  //11. IIC_Stop                  ; 结束信号
22
23   return uRev_Register_Data;
24
25 }

  

  完成读写函数的编写后,那么我们就可测试传感是否通信成功,具体可检测的方法有挺多,最直观的是用示波器察看波形,若在读写函数发送器件地址后,有接下来的发送寄存器地址和数据等操作波形的话,即有返回到ACK,则单片机与传感通信成功。这里再说一个不需要用到的示波器的操作,同样的,在读写函数发送器件地址后,用Uart发送功能,在串口助手上打印一个数,若有数显示,则没有return 跳出函数,证明SDA的电平被器件拉低,也是有ACK返回的情况,则单片机于传感通信成功。

4.4.5 修改时间

  我们在读取时间之前需要对时间进行一个手动输入的方法去给予时钟模块一个初始值,这样才能开始计时,这个函数可以当作修改时间的功能,具体代码可以自行编写,我这里的代码就写成全部修改。

 1 void vSen_DS3231_Modify_Time(IIC_HandleTypedef * iicHandle, uint8_t uyear, uint8_t uMouth, uint8_t uData, uint8_t uDay, uint8_t uHour, uint8_t uMinutes, uint8_t uSeconds)
 2 {
 3
 4   vSen_DS3231_Write_Bytes(iicHandle, Year_Register,    Byte_Transform_BCD(uyear));                //修改年份
 5
 6   vSen_DS3231_Write_Bytes(iicHandle, Month_Register,   Byte_Transform_BCD(uMouth));               //修改月份
 7
 8   vSen_DS3231_Write_Bytes(iicHandle, Date_Register,    Byte_Transform_BCD(uData));                //修改日期
 9
10   vSen_DS3231_Write_Bytes(iicHandle, Day_Register,     Byte_Transform_BCD(uDay));                 //修改星期
11
12   vSen_DS3231_Write_Bytes(iicHandle, Hour_Register,    Byte_Transform_BCD(uHour));                //修改小时
13
14   vSen_DS3231_Write_Bytes(iicHandle, Minutes_Register, Byte_Transform_BCD(uMinutes));             //修改分钟
15
16   vSen_DS3231_Write_Bytes(iicHandle, Seconds_Register, Byte_Transform_BCD(uSeconds));             //修改秒钟
17
18 }

4.4.6 获取时间

  获取时间就直接读取相应的寄存器获取数值即可,最后将其输出就行。

 1 void vSen_DS3231_Get_Times(IIC_HandleTypedef * iicHandle, DS3231_HandleTypeDef * hDS3231)
 2 {
 3   hDS3231->uyear    =  BCD_Transform_Byte(vSen_DS3231_Read_Bytes(iicHandle, Year_Register));      //获取年份
 4
 5   hDS3231->uMouth   =  BCD_Transform_Byte(vSen_DS3231_Read_Bytes(iicHandle, Month_Register));     //获取月份
 6
 7   hDS3231->uData    =  BCD_Transform_Byte(vSen_DS3231_Read_Bytes(iicHandle, Date_Register));      //获取日期
 8
 9   hDS3231->uDay     =  BCD_Transform_Byte(vSen_DS3231_Read_Bytes(iicHandle, Day_Register));       //获取星期
10
11   hDS3231->uHour    =  BCD_Transform_Byte(vSen_DS3231_Read_Bytes(iicHandle, Hour_Register));      //获取小时
12
13   hDS3231->uMinutes =  BCD_Transform_Byte(vSen_DS3231_Read_Bytes(iicHandle, Minutes_Register));   //获取分钟
14
15   hDS3231->uSeconds =  BCD_Transform_Byte(vSen_DS3231_Read_Bytes(iicHandle, Seconds_Register));   //获取秒钟
16
17 }

4.5 成果展示

  因为我这里用的是数码管显示,但数码管格数不够,就只显示了时分秒三个数值。

5. 结尾

  DS3231模块已经成功驱动了,毕竟这个模块是比较容易的一个IIC传感,基本不用配置什么寄存器,直接写入再直接读取即可,算是一个IIC小入门难度,不过其他传感也大同小异,都非常类似。

  对STM8的I2C传感模块驱动讲解到这里结束,感谢各位看官的点击。

  如果觉得有所收获请点下推荐,若认为该博客中存在错误的说明或者对博客中某方面有疑问请留言。

作 者:浩宇99?
出 处:https://www.cnblogs.com/zhenghaoyu/p/10841542.html版权声明:本文原创发表于 博客园,欢迎转载,但未经作者同意必须保留此段声明,且在文章页面明显位置给出原文连接,否则视为侵权。

原文地址:https://www.cnblogs.com/zhenghaoyu/p/10841542.html

时间: 2024-10-13 16:40:11

基于STM8的IIC协议--实例篇--时钟模块(DS3231)读取的相关文章

张高兴的 Windows 10 IoT 开发笔记:RTC 时钟模块 DS3231

原文:张高兴的 Windows 10 IoT 开发笔记:RTC 时钟模块 DS3231 GitHub:https://github.com/ZhangGaoxing/windows-iot-demo/tree/master/DS3231 注意:不包含闹钟设置

基于EasyDarwin的实现无人机远程视频传输--RTSP协议分析篇

申明该文章参考了http://blog.csdn.net/haolipengzhanshen/article/details/50802081 的文章,在这里标示感谢! 这篇文章主要从几个方面分析EasyDarwin的RTSP内容 RTSP协议概述 wireshark抓包实例分析 一次完整RTSP的交互流程 EasyDarwin项目代码中 RTSP的初始化 EasyDarwin项目代码中 RTSP请求的处理过程 如果你是只想实现视频流的传输,对转发服务器没有太大要求,建议只要研究EasyDarw

基于Python的接口测试框架实例

文章来源:http://www.jb51.net/article/96481.htm 下面小编就为大家带来一篇基于Python的接口测试框架实例.小编觉得挺不错的,现在就分享给大家,也给大家做个参考.一起跟随小编过来看看吧 背景 最近公司在做消息推送,那么自然就会产生很多接口,测试的过程中需要调用接口,我就突然觉得是不是可以自己写一个测试框架? 说干就干,由于现有的接口测试工具Jmeter.SoupUI等学习周期有点长,干脆自己写一个吧,不求人,所有功能自己都能一清二楚. 当然,写工具造轮子只是

FPGA基础设计(四):IIC协议

很多数字传感器.数字控制的芯片(DDS.串行ADC.串行DAC)都是通过IIC总线来和控制器通信的.不过IIC协议仍然是一种慢速的通信方式,标准IIC速率为100kbit/s,快速模式速率为400kbit/s.本文致力于讲述如何用计数器控制和分频时钟控制两种方式完成IIC的读写操作. IIC协议 ??IIC协议是一种多机通讯,由SDA数据线和SCL时钟线构成串行总线,所有的IIC设备都可以挂载到总线上,但每个设备都有唯一的设备读地址和设备写地址.在使用IIC作为数字接口的芯片datasheet中

HTTP协议——深入篇

HTTP协议——深入篇 我们知道HTTP是由请求和响应两部分组成当你在Web浏览器中输入一个URL时,浏览器将根据你的要求创建并发送请求,该请求包含所输入的URL以及一些与浏览器本身相关的信息.当服务器收到这个请求时将返回一个响应,该响应包括与该请求相关的信息以及位于指定URL(如果有的话)的数据.直到浏览器解析该响应并显示出网页(或其他资源)为止. HTTP请求 HTTP请求的格式如下所示: <request-line><注释--请求行--> <headers><

使用Myeclipse 8.5开发基于JAX-WS的Web service实例

使用Myeclipse 8.5开发基于JAX-WS的Web service实例  本文为Web service 开发入门篇,主要介绍在Myeclipse 8.5环境下开发Web service的服务程序和客户端程序的基本流程.  在Weblogic 10.3.4 中部署Web service服务.   开发环境如下:  JAVA IDE: Myeclipse 8.5 开发Web service服务程序,需要了解以下相关内容, WSDL, SOAP, XML.这些是组成Web service 的基

stm8的IIC库的使用

一.前言 stm8是一款低功耗的MCU芯片,它具备stm32库函数和资源丰富的优势.也同时具有价格便宜,低功耗的特点.在一些项目中,能起到很好的作用.下面我介绍一下stm8的IIC硬件库函数驱动代码及实现. 二.IIC基本操作 iic基本操作分为读操作和写操作,这两个操作就可以对设备进行一些基本的操作了.还要知道设备的地址就ok了,具体的IIC时序大家可以上网看看. 三.实现代码 1.先要启动IIC的时钟,保证可以正常工作. CLK_PeripheralClockConfig (CLK_Peri

IIC协议解释(转)

IIC协议解释 推荐资源: http://m.elecfans.com/article/574049.html       and       https://blog.csdn.net/firefly_cjd/article/details/51921129  (动态图讲解) (1)概述 I2C(Inter-Integrated Circuit BUS) 集成电路总线,该总线由NXP(原PHILIPS)公司设计,多用于主控制器和从器件间的主从通信,在小数据量场合使用,传输距离短,任意时刻只能有

IIC协议理解(转)

目录 IIC协议理解(转) 个人小结记录 (记一下这个就够了) 以下为转载记录 概述 概述 输出级 主设备与从设备 速率 时序 空闲状态 起始位与停止位 数据的有效性 数据的传送 工作过程 主设备向从设备发送数据 主控器读取数据的过程 以C语言理解IIC title: IIC协议理解(转) date: 2019/1/28 17:50:13 toc: true --- IIC协议理解(转) 原文地址 https://www.cnblogs.com/BitArt/archive/2013/05/28