当前位置:   article > 正文

基于stm32f103zet6的DS1302学习_stm32f103zet6 ds1302

stm32f103zet6 ds1302

由于硬件出了问题,也就是外部低速晶振没用,震不起来,然后查看了网上的帖子,STM32的RTC果然口碑不怎么样,所以果断换DS1302,在移植的过程中还算顺利,记录下来吧,也算对自己的总结吧!

1、所谓的DS1302


这里面也指明了简单的SPI协议

然后就直接上代码了,注释很详细的!基本实现单行注释

1、主函数main

  1. #include "stm32f10x.h"
  2. #include "SysTick.h"
  3. #include "Delay.h"
  4. #include "Usart.h"
  5. #include "stdio.h"
  6. #include "DS1302.h"
  7. /*******由于没有做外设测试的程序是:按键PA0仅一个LED灯 ******/
  8. /*******由于没有做外设测试的程序是:串口采用的是PA9->(T<->T),PA9->(R<->R)*****/
  9. /*******由于没有做外设测试的程序是:ds1302clk <-> PA4
  10. ds1302dat <-> PA5
  11. ds1302rst <-> PA6 *****/
  12. extern uint8_t write[];
  13. extern uint8_t read[];
  14. int main(void)
  15. {
  16. //uint8_t time[] = {0,0,12,11,11,1,13};
  17. //初始化系统定时器
  18. SysTick_Init();
  19. USART1_Config();
  20. ds1302_GPIO_Configuration(); //一定别忘记了配置GPIO
  21. printf("\r\n ("__DATE__ " - " __TIME__ ") \r\n");
  22. //ds1302_init(write,time); //设置1302的初始时间
  23. ds1302_data(read);
  24. while(1); //这个得加,不然的话会有尾巴是乱码!!!
  25. }

  1. /*******************************************************************************
  2. ** 文件名称 : ds1302.c
  3. ** 编译环境 : RealView MDK-ARM 4.53
  4. ** 文件作者 :
  5. ** 功能说明 : ds1302相关函数定义
  6. ** 软件协议 :模拟SPI协议
  7. ** 简介 :clk上升沿发送数据,clk下降沿接收数据
  8. :typedef unsigned char uint8_t;
  9. **
  10. **
  11. **
  12. *******************************************************************************/
  13. /* 包含头文件 *****************************************************************/
  14. #include "ds1302.h"
  15. #include "stdio.h"
  16. #include "Delay.h"
  17. uint8_t read[] = {0x81,0x83,0x85,0x87,0x89,0x8b,0x8d};//读秒、分、时、日、月、周、年的寄存器地址
  18. uint8_t write[] = {0x80,0x82,0x84,0x86,0x88,0x8a,0x8c};//写秒、分、时、日、月、周、年的寄存器地址
  19. /*PA4、PA6为输出*/
  20. /*PA5配置为开漏模式,此模式下能够实现真正的双向IO口*/
  21. void ds1302_GPIO_Configuration(void)
  22. {
  23. GPIO_InitTypeDef GPIO_InitStruct;
  24. GPIO_InitStruct.GPIO_Mode = GPIO_Mode_Out_PP;
  25. GPIO_InitStruct.GPIO_Pin = ds1302clk|ds1302rst; //clk、rst配置为输出
  26. GPIO_InitStruct.GPIO_Speed = GPIO_Speed_50MHz;
  27. GPIO_Init(GPIOA, &GPIO_InitStruct);
  28. GPIO_InitStruct.GPIO_Mode = GPIO_Mode_Out_OD; //IO配置为双向
  29. GPIO_InitStruct.GPIO_Pin = ds1302dat;
  30. GPIO_Init(GPIOA, &GPIO_InitStruct);
  31. }
  32. void write_1302byte(uint8_t dat) //写一个字节的数据sck上升沿写数据
  33. {
  34. uint8_t i = 0;
  35. GPIO_ResetBits(GPIOA,ds1302clk); //ds1302clk=0
  36. Delay_us(2);//延时大约2us
  37. for(i = 0;i < 8;i ++)
  38. {
  39. GPIO_ResetBits(GPIOA,ds1302clk); //ds1302clk=0;
  40. if(dat&0x01)
  41. GPIO_SetBits(GPIOA,ds1302dat);
  42. else //ds1302dat=(dat&0x01)
  43. GPIO_ResetBits(GPIOA,ds1302dat);
  44. Delay_us(2);
  45. GPIO_SetBits(GPIOA,ds1302clk); //发送一位数据,clk上升沿,//ds1302clk=1
  46. dat >>= 1;
  47. Delay_us(1);
  48. }
  49. }
  50. uint8_t read_1302(uint8_t add) //读数据
  51. {
  52. uint8_t i=0;
  53. uint8_t Return_dat=0x00;
  54. GPIO_ResetBits(GPIOA,ds1302rst); //ds1302rst=0;
  55. GPIO_ResetBits(GPIOA,ds1302clk); //ds1302clk=0;
  56. Delay_us(3); //略微延时2us
  57. GPIO_SetBits(GPIOA,ds1302rst); //ds1302rst=1;
  58. Delay_us(3); //时间要大约3us
  59. write_1302byte(add); //先写寄存器的地址
  60. for(i=0;i<8;i++)
  61. {
  62. GPIO_SetBits(GPIOA,ds1302clk); //ds1302clk=1;
  63. Return_dat >>= 1;
  64. GPIO_ResetBits(GPIOA,ds1302clk); //ds1302clk=0;//拉低时钟线,以便于数据的读入
  65. if(GPIO_ReadInputDataBit(GPIOA,ds1302dat)==1) //数据线此时为高电平
  66. {Return_dat = Return_dat|0x80;}
  67. }
  68. Delay_us(1);
  69. GPIO_ResetBits(GPIOA,ds1302rst); //ds1302rst=0;释放总线
  70. return Return_dat;
  71. }
  72. void write_1302(uint8_t add,uint8_t dat) //向指定寄存器写入一个字节的数据
  73. {
  74. GPIO_ResetBits(GPIOA,ds1302rst); //只有在rst为高电平的时候才能进行数据传输
  75. GPIO_ResetBits(GPIOA,ds1302clk); //只有clk为低电平的时候,rst才能被置为高电平
  76. //ds1302rst=0;
  77. //ds1302clk=0;
  78. Delay_us(1); //略微延时
  79. GPIO_SetBits(GPIOA,ds1302rst); //clk = 0之后,这里将rst拉高,准备传送数据
  80. //ds1302rst=1;
  81. Delay_us(2); //时间大约2us
  82. write_1302byte(add); //先发地址
  83. write_1302byte(dat); //然后发数据
  84. GPIO_ResetBits(GPIOA,ds1302rst); //这里释放总线
  85. GPIO_ResetBits(GPIOA,ds1302clk); //拉低clk,以备下一次数据发送
  86. //ds1302clk=0;
  87. //ds1302rst=0;
  88. Delay_us(1);
  89. }
  90. void ds1302_init(uint8_t *write,uint8_t *time) //初始化1302
  91. {
  92. uint8_t i=0,j=0;
  93. write_1302(0x8e,0x00); //关闭写保护
  94. for(i=0;i<7;i++) //十进制转BCD码
  95. {
  96. j=time[i]%10; //个位数部分
  97. time[i]=(time[i]/10)*16+j;
  98. }
  99. for(i=0;i<7;i++) //进行对时
  100. {
  101. write_1302(write[i],time[i]); //在对应寄存器上写入对应的十六进制数据
  102. }
  103. write_1302(0x8e,0x80); //打开写保护
  104. }
  105. void ds1302_data(uint8_t *read) //处理数据并通过串口打印
  106. {
  107. uint8_t i=0,g[7],time[7];
  108. // static uint8_t s = 1;
  109. for(i=0;i<7;i++)
  110. {
  111. time[i]=read_1302(read[i]); //读数据已经完成
  112. }
  113. for(i=0;i<7;i++)
  114. {
  115. g[i]=time[i]%16; //秒个位数据:BCD转十进制
  116. time[i]=time[i]/16; //秒十位数据
  117. }
  118. //此时已转换成10进制数,g[i]里面存放的是秒分时日月周年的各个位数据
  119. //而此时的time[i]存放的是十位数据
  120. //if(s != (time[0]+g[0]))
  121. printf("20%d%d年%d%d月%d%d日%d%d:%d%d:%d%d 星期%d\r\n",time[6],g[6],time[4],g[4],time[3],g[3],time[2],g[2],time[1],g[1],time[0],g[0],g[5]);
  122. //s = time[0]+g[0];
  123. }
就是这样的简单,然后给出一张效果图


声明:本文内容由网友自发贡献,不代表【wpsshop博客】立场,版权归原作者所有,本站不承担相应法律责任。如您发现有侵权的内容,请联系我们。转载请注明出处:https://www.wpsshop.cn/w/笔触狂放9/article/detail/460806
推荐阅读
相关标签
  

闽ICP备14008679号