第30章      RL-TCPnet之SNTP网络时间获取

本章节为大家讲解RL-TCPnet的SNTP应用,学习本章节前,务必要优先学习第29章的NTP基础知识。有了这些基础知识之后,再搞本章节会有事半功倍的效果。

本章教程含STM32F407开发板和STM32F429开发板。

30.1  初学者重要提示

30.2  可用的NTP服务器

30.3  SNTP函数

30.4  SNTP配置说明(Net_Config.c)

30.5  SNTP调试说明(Net_Debug.c)

30.6  网络调试助手和板子的操作步骤

30.7  实验例程说明(裸机)

30.8  实验例程说明(RTX)

30.9  总结

30.1  初学者重要提示

  1. 学习本章节前,务必保证已经学习了第29章的基础知识。
  2. 本章相对比较简单,测试本章节的例子,务必将其接到能够联网的路由器或者交换机上

30.2  可用的NTP服务器

国内可用的NTP服务器已经在帖子:http://bbs.armfly.com/read.php?tid=31397 里面进行了简单的总结,我们这里使用IP地址为182.16.3.162的NTP服务器。

30.3  SNTP函数

涉及到SNTP的,仅有如下一个函数:

  • sntp_get_time

关于这个函数的讲解及其使用方法可以看教程第 3 章 3.4 小节里面说的参考资料 rlarm.chm 文件:

注意,这个函数不支持重入,也就是不支持多任务调用。

30.3.1   函数sntp_get_time

函数原型:

  1. BOOL sntp_get_time (
  2.  
  3. U8* ipadr, /* NTP/SNTP服务器IP地址 */
  4.  
  5. void (*cbfunc)( /* 回调函数,接收到NTP消息会触发 */
  6.  
  7. U32 utc_time) ); /* 接收到的UNIX时间戳,从1970.1.1开始所经历的秒数 */

函数描述:

函数sntp_get_time用于从NTP服务器获得UNIX时间戳,这个函数支持单播和广播两种模式。单播方式下,通过此函数给远程NTP服务器发送获取时间消息。广播模式下,将打开UDP Socket接收NTP广播消息,如果局域网内有NTP服务器,可以采用这种模式。(广播和单播模式是在Net_Config.c文件中设置的,如果没有选择广播Broadcast模式,就表示单播,否则表示广播。

  1. 第1个参数是NTP服务器的IP地址。

    • 单播模式,这个参数就是远程NTP服务器的IP地址。
    • 广播模式,这个参数是局域网内NTP服务器的IP地址,如果用户设置了指定的IP地址,那么将仅接收此服务器的消息,其它服务器的消息忽略。如果此IP地址被设置为0.0.0.0,那么将接收局域网内任何NTP服务器的消息。
  2. 第2个参数是回调函数,回调函数有一个参数,这个参数utc_time代表的含义如下:
    • 调用此函数返回的UNIX时间戳,从1970.1.1开始所经历的秒数,如果此数值是0的话(数值0被保留用于表示返回失败),表示获取失败。
  3. 返回值有以下两种,返回__TURE,单播模式下表示SNTP消息发送成功,广播模式下表示UDP Socket打开成功。返回__FALSE,单播模式下表示发送失败,广播模式下表示UDP Socket打开失败。

使用这个函数要注意以下问题:

  1. 如果用户将第1个参数设置为NULL的话,那么将使用Net_Congfig.c文件中设置的NTP服务器地址。

使用举例:

  1. /*
  2.  
  3. *********************************************************************************************************
  4.  
  5. * 用于本文件的调试
  6.  
  7. *********************************************************************************************************
  8.  
  9. */
  10.  
  11. #if 1
  12.  
  13. #define printf_debug printf
  14.  
  15. #else
  16.  
  17. #define printf_debug(...)
  18.  
  19. #endif
  20.  
  21. /*
  22.  
  23. **********************************************************************************************************
  24.  
  25. 变量
  26.  
  27. **********************************************************************************************************
  28.  
  29. */
  30.  
  31. U8 ntp_server[] = {,,,}; /* SNTP服务器地址 */
  32.  
  33. /*
  34.  
  35. *********************************************************************************************************
  36.  
  37. * 函 数 名: time_cback
  38.  
  39. * 功能说明: SNTP获取时间回到函数
  40.  
  41. * 形 参: 无
  42.  
  43. * 返 回 值: 无
  44.  
  45. *********************************************************************************************************
  46.  
  47. */
  48.  
  49. static void time_cback (uint32_t time)
  50.  
  51. {
  52.  
  53. struct tm *t_tm;
  54.  
  55. if (time == )
  56.  
  57. {
  58.  
  59. printf_debug ("错误, 服务器未响应或者网络状态比较差\r\n");
  60.  
  61. }
  62.  
  63. else
  64.  
  65. {
  66.  
  67. /* 由于是格林尼治时间,所以北京时间要加8个小时 */
  68.  
  69. time += **;
  70.  
  71. /* 将获取的UNIX时间戳转换成年月日时分秒的格式 */
  72.  
  73. t_tm = localtime((unsigned int *)&time);
  74.  
  75. /* 格林尼治时间是从1900年开始的,这要加上 */
  76.  
  77. t_tm->tm_year += ;
  78.  
  79. /* 获取的月份范围是0-11,这里要加1 */
  80.  
  81. t_tm->tm_mon += ;
  82.  
  83. printf_debug ("UNIX时间戳:%d 日期:%02d/%02d/%02d 时间:%02d:%02d:%02d\r\n",
  84.  
  85. time,
  86.  
  87. t_tm->tm_year,
  88.  
  89. t_tm->tm_mon,
  90.  
  91. t_tm->tm_mday,
  92.  
  93. t_tm->tm_hour,
  94.  
  95. t_tm->tm_min,
  96.  
  97. t_tm->tm_sec);
  98.  
  99. }
  100.  
  101. }
  102.  
  103. /*
  104.  
  105. *********************************************************************************************************
  106.  
  107. * 函 数 名: get_time
  108.  
  109. * 功能说明: 从SNTP服务器获取当前时间
  110.  
  111. * 形 参: 无
  112.  
  113. * 返 回 值: 无
  114.  
  115. *********************************************************************************************************
  116.  
  117. */
  118.  
  119. static void get_time (void)
  120.  
  121. {
  122.  
  123. if (sntp_get_time((U8 *)&ntp_server[], &time_cback) == __TRUE)
  124.  
  125. {
  126.  
  127. printf_debug ("SNTP请求已经发送成功\r\n");
  128.  
  129. }
  130.  
  131. else
  132.  
  133. {
  134.  
  135. printf_debug ("失败, SNTP未就绪或者参数错误\r\n");
  136.  
  137. }
  138.  
  139. }
  140.  
  141. /*
  142.  
  143. *********************************************************************************************************
  144.  
  145. * 函 数 名: tcpnet_poll
  146.  
  147. * 功能说明: 使用TCPnet必须要一直调用的函数
  148.  
  149. * 形 参: 无
  150.  
  151. * 返 回 值: 无
  152.  
  153. *********************************************************************************************************
  154.  
  155. */
  156.  
  157. void tcpnet_poll(void)
  158.  
  159. {
  160.  
  161. if(bsp_CheckTimer())
  162.  
  163. {
  164.  
  165. bsp_LedToggle();
  166.  
  167. /* 此函数坚决不可以放在中断里面跑 */
  168.  
  169. timer_tick ();
  170.  
  171. }
  172.  
  173. main_TcpNet ();
  174.  
  175. }
  176.  
  177. /*
  178.  
  179. *********************************************************************************************************
  180.  
  181. * 函 数 名: TCPnetTest
  182.  
  183. * 功能说明: TCPnet应用
  184.  
  185. * 形 参: 无
  186.  
  187. * 返 回 值: 无
  188.  
  189. *********************************************************************************************************
  190.  
  191. */
  192.  
  193. void TCPnetTest(void)
  194.  
  195. {
  196.  
  197. /* 初始化网络协议栈 */
  198.  
  199. init_TcpNet ();
  200.  
  201. /* 创建一个周期是100ms的软定时器 */
  202.  
  203. bsp_StartAutoTimer(, );
  204.  
  205. /* 再创建一个周期是1000ms的软定时器 */
  206.  
  207. bsp_StartAutoTimer(, );
  208.  
  209. while ()
  210.  
  211. {
  212.  
  213. /* TCP轮询 */
  214.  
  215. tcpnet_poll();
  216.  
  217. /* 每秒从NTP服务器获取一次时间 */
  218.  
  219. if(bsp_CheckTimer())
  220.  
  221. {
  222.  
  223. get_time();
  224.  
  225. }
  226.  
  227. }
  228.  
  229. }

30.4 SNTP配置说明(Net_Config.c)

(本章节配套例子的配置与本小节的说明相同)

RL-TCPnet的配置工作是通过配置文件Net_Config.c实现。在MDK工程中打开文件Net_Config.c,可以看到下图所示的工程配置向导:

RL-TCPnet要配置的选项非常多,我们这里把几个主要的配置选项简单介绍下。

System Definitions

(1)Local Host Name

局域网域名。

这里起名为armfly,使用局域网域名限制为15个字符。

(2)Memory Pool size

参数范围1536-262144字节。

内存池大小配置,单位字节。另外注意一点,配置向导这里显示的单位是字节,如果看原始定义,MDK会做一个自动的4字节倍数转换,比如我们这里配置的是8192字节,那么原始定义是#define MEM_SIZE  2048,也就是8192/4 = 2048。

(3)Tick Timer interval

可取10,20,25,40,50,100,200,单位ms。

系统滴答时钟间隔,也就是网络协议栈的系统时间基准,默认情况下,取值100ms。

Ethernet Network Interface

以太网接口配置,勾选了此选项就可以配置了,如果没有使能DHCP的话,将使用这里配置的固定IP

(1) MAC Address

局域网内可以随意配置,只要不跟局域网内其它设备的MAC地址冲突即可。

(2)IP Address

IP地址。

(3)Subnet mask

子网掩码。

(4)Default Gateway

默认网关。

Ethernet Network Interface

以太网接口配置,这个配置里面还有如下两项比较重要的配置需要说明。

(1)NetBIOS Name Service

NetBIOS局域网域名服务,这里打上对勾就使能了。这样我们就可以通过前面配置的Local Host Name局域网域名进行访问,而不需要通过IP地址访问了。

(2)Dynaminc Host Configuration

即DHCP,这里打上对勾就使能了。使能了DHCP后,RL-TCPnet就可以从外接的路由器上获得动态IP地址。

UDP Sockets

UDP Sockets配置,打上对勾就使能了此项功能

(1) Number of UDP Sockets

用于配置可创建的UDP Sockets数量,这里配置了5个。

范围1 – 20。

TCP Sockets

TCP Sockets配置,打上对勾就使能了此项功能

(1) Number of TCP Sockets

用于配置可创建的TCP Sockets数量。

(2)Number of Retries

范围0-20。

用于配置重试次数,TCP数据传输时,如果在设置的重试时间内得不到应答,算一次重试失败,这里就是配置的最大重试次数。

(3)Retry Timeout in seconds

范围1-10,单位秒。

重试时间。如果发送的数据在重试时间内得不到应答,将重新发送数据。

(4)Default Connect Timeout in seconds

范围1-600,单位秒。

用于配置默认的保持连接时间,即我们常说的Keep Alive时间,如果时间到了将断开连接。常用于HTTP Server,Telnet Server等。

(5)Maximum Segment Size

范围536-1460,单位字节。

MSS定义了TCP数据包能够传输的最大数据分段。

(6)Receive Window Size

范围536-65535,单位字节。

TCP接收窗口大小。

SNTP Client

SNTP配置,打上对勾就使能了此项功能

(1) Broadcast Mode

打上对勾表示使能广播模式,不选择表示单播模式。

(2)NTP Server

这里是NTP服务器的IP地址。

实际应用中,这两个选项的作用看本章30.3.1小节的函数sntp_get_time即可。

30.5 SNTP调试说明(Net_Debug.c)

(重要说明,RL-TCPnet的调试是通过串口打印出来的)

RL-TCPnet的调试功能是通过配置文件Net_Debug.c实现。在MDK工程中打开文件Net_Debug.c,可以看到如下图所示的工程配置向导:

Print Time Stamp

勾选了此选项的话,打印消息时,前面会附带时间信息。

其它所有的选项

默认情况下,所有的调试选项都关闭了,每个选项有三个调试级别可选择,这里我们以SNTP Debug为例,点击下拉列表,可以看到里面有Off,Errors only和Full debug三个调试级别可供选择,每个调试选项里面都是这三个级别。

Off:表示关闭此选项的调试功能。

Errors only:表示仅在此选项出错时,将其错误打印出来。

Full debug:表示此选项的全功能调试。

下面是对SNTP Debug配置为Full debug时,打印出来的消息(NTP服务器采用的182.16.3.162):

30.6 板子的操作步骤

本章的操作相对比较简单,用户务必将板子连接到能够联网的路由器或者交换机上。串口会每秒打印一次获取的UNIX时间戳(波特率115200,数据位8,奇偶校验位无,停止位1):

30.7 实验例程说明(裸机)

30.7.1 STM32F407开发板实验

配套例子:

V5-1040_RL-TCPnet实验SNTP应用(裸机)

实验目的:

  1. 学习RL-TCPnet的SNTP使用。

实验内容:

  1. 务必将网线接到能够联网的路由器或者交换机上面测试,因为本实验要用到外网。
  2. 本实验串口每秒打印一次从NTP服务器获取的时间。
  3. 国内免费稳定的SNTP服务器很少,当前使用的这个时好时坏,如果大家测试不成功的话也是正常的。

实验操作:

详见本章节30.6小节。

配置向导文件设置(Net_Config.c):

详见本章节30.4小节。

调试文件设置(Net_Debug.c):

详见本章节30.5小节。

程序设计:

主函数初始化

在main.c文件实现:

  1. /*
  2.  
  3. *********************************************************************************************************
  4.  
  5. * 函 数 名: main
  6.  
  7. * 功能说明: 标准c程序入口。
  8.  
  9. * 形 参: 无
  10.  
  11. * 返 回 值: 无
  12.  
  13. *********************************************************************************************************
  14.  
  15. */
  16.  
  17. int main (void)
  18.  
  19. {
  20.  
  21. /* 初始化外设 */
  22.  
  23. bsp_Init();
  24.  
  25. /* 进入RL-TCPnet测试函数 */
  26.  
  27. TCPnetTest();
  28.  
  29. }

硬件外设初始化

硬件外设的初始化是在 bsp.c 文件实现:

  1. /*
  2.  
  3. *********************************************************************************************************
  4.  
  5. * 函 数 名: bsp_Init
  6.  
  7. * 功能说明: 初始化所有的硬件设备。该函数配置CPU寄存器和外设的寄存器并初始化一些全局变量。只需要调用一次
  8.  
  9. * 形 参:无
  10.  
  11. * 返 回 值: 无
  12.  
  13. *********************************************************************************************************
  14.  
  15. */
  16.  
  17. void bsp_Init(void)
  18.  
  19. {
  20.  
  21. /*
  22.  
  23. 由于ST固件库的启动文件已经执行了CPU系统时钟的初始化,所以不必再次重复配置系统时钟。
  24.  
  25. 启动文件配置了CPU主时钟频率、内部Flash访问速度和可选的外部SRAM FSMC初始化。
  26.  
  27. 系统时钟缺省配置为168MHz,如果需要更改,可以修改 system_stm32f4xx.c 文件
  28.  
  29. */
  30.  
  31. /* 优先级分组设置为4,可配置0-15级抢占式优先级,0级子优先级,即不存在子优先级。*/
  32.  
  33. NVIC_PriorityGroupConfig(NVIC_PriorityGroup_4);
  34.  
  35. bsp_InitUart(); /* 初始化串口 */
  36.  
  37. bsp_InitKey(); /* 初始化按键变量(必须在 bsp_InitTimer() 之前调用) */
  38.  
  39. bsp_InitLed(); /* 初始LED指示灯端口 */
  40.  
  41. bsp_InitTimer(); /* 初始化滴答定时器 */
  42.  
  43. }

RL-TCPnet功能测试

这里专门创建了一个app_tcpnet_lib.c文件用于RL-TCPnet功能的测试,此文件主要实现从远程NTP服务器获取当前日期和时间,精度到秒。

  1. /*
  2.  
  3. *********************************************************************************************************
  4.  
  5. * 用于本文件的调试
  6.  
  7. *********************************************************************************************************
  8.  
  9. */
  10.  
  11. #if 1
  12.  
  13. #define printf_debug printf
  14.  
  15. #else
  16.  
  17. #define printf_debug(...)
  18.  
  19. #endif
  20.  
  21. /*
  22.  
  23. **********************************************************************************************************
  24.  
  25. 变量
  26.  
  27. **********************************************************************************************************
  28.  
  29. */
  30.  
  31. U8 ntp_server[] = {,,,}; /* SNTP服务器地址 */
  32.  
  33. /*
  34.  
  35. *********************************************************************************************************
  36.  
  37. * 函 数 名: time_cback
  38.  
  39. * 功能说明: SNTP获取时间回到函数
  40.  
  41. * 形 参: 无
  42.  
  43. * 返 回 值: 无
  44.  
  45. *********************************************************************************************************
  46.  
  47. */
  48.  
  49. static void time_cback (uint32_t time)
  50.  
  51. {
  52.  
  53. struct tm *t_tm;
  54.  
  55. if (time == )
  56.  
  57. {
  58.  
  59. printf_debug ("错误, 服务器未响应或者网络状态比较差\r\n");
  60.  
  61. }
  62.  
  63. else
  64.  
  65. {
  66.  
  67. /* 由于是格林尼治时间,所以北京时间要加8个小时 */
  68.  
  69. time += **;
  70.  
  71. /* 将获取的UNIX时间戳转换成年月日时分秒的格式 */
  72.  
  73. t_tm = localtime((unsigned int *)&time);
  74.  
  75. /* 格林尼治时间是从1900年开始的,这要加上 */
  76.  
  77. t_tm->tm_year += ;
  78.  
  79. /* 获取的月份范围是0-11,这里要加1 */
  80.  
  81. t_tm->tm_mon += ;
  82.  
  83. printf_debug ("UNIX时间戳:%d 日期:%02d/%02d/%02d 时间:%02d:%02d:%02d\r\n",
  84.  
  85. time,
  86.  
  87. t_tm->tm_year,
  88.  
  89. t_tm->tm_mon,
  90.  
  91. t_tm->tm_mday,
  92.  
  93. t_tm->tm_hour,
  94.  
  95. t_tm->tm_min,
  96.  
  97. t_tm->tm_sec);
  98.  
  99. }
  100.  
  101. }
  102.  
  103. /*
  104.  
  105. *********************************************************************************************************
  106.  
  107. * 函 数 名: get_time
  108.  
  109. * 功能说明: 从SNTP服务器获取当前时间
  110.  
  111. * 形 参: 无
  112.  
  113. * 返 回 值: 无
  114.  
  115. *********************************************************************************************************
  116.  
  117. */
  118.  
  119. static void get_time (void)
  120.  
  121. {
  122.  
  123. if (sntp_get_time((U8 *)&ntp_server[], &time_cback) == __TRUE)
  124.  
  125. {
  126.  
  127. printf_debug ("SNTP请求已经发送成功\r\n");
  128.  
  129. }
  130.  
  131. else
  132.  
  133. {
  134.  
  135. printf_debug ("失败, SNTP未就绪或者参数错误\r\n");
  136.  
  137. }
  138.  
  139. }
  140.  
  141. /*
  142.  
  143. *********************************************************************************************************
  144.  
  145. * 函 数 名: tcpnet_poll
  146.  
  147. * 功能说明: 使用TCPnet必须要一直调用的函数
  148.  
  149. * 形 参: 无
  150.  
  151. * 返 回 值: 无
  152.  
  153. *********************************************************************************************************
  154.  
  155. */
  156.  
  157. void tcpnet_poll(void)
  158.  
  159. {
  160.  
  161. if(bsp_CheckTimer())
  162.  
  163. {
  164.  
  165. bsp_LedToggle();
  166.  
  167. /* 此函数坚决不可以放在中断里面跑 */
  168.  
  169. timer_tick ();
  170.  
  171. }
  172.  
  173. main_TcpNet ();
  174.  
  175. }
  176.  
  177. /*
  178.  
  179. *********************************************************************************************************
  180.  
  181. * 函 数 名: TCPnetTest
  182.  
  183. * 功能说明: TCPnet应用
  184.  
  185. * 形 参: 无
  186.  
  187. * 返 回 值: 无
  188.  
  189. *********************************************************************************************************
  190.  
  191. */
  192.  
  193. void TCPnetTest(void)
  194.  
  195. {
  196.  
  197. /* 初始化网络协议栈 */
  198.  
  199. init_TcpNet ();
  200.  
  201. /* 创建一个周期是100ms的软定时器 */
  202.  
  203. bsp_StartAutoTimer(, );
  204.  
  205. /* 再创建一个周期是1000ms的软定时器 */
  206.  
  207. bsp_StartAutoTimer(, );
  208.  
  209. while ()
  210.  
  211. {
  212.  
  213. /* TCP轮询 */
  214.  
  215. tcpnet_poll();
  216.  
  217. /* 每秒从NTP服务器获取一次时间 */
  218.  
  219. if(bsp_CheckTimer())
  220.  
  221. {
  222.  
  223. get_time();
  224.  
  225. }
  226.  
  227. }
  228.  
  229. }

30.7.2 STM32F429开发板实验

配套例子:

V6-1040_RL-TCPnet实验SNTP应用(裸机)

实验目的:

  1. 学习RL-TCPnet的SNTP使用。

实验内容:

  1. 务必将网线接到能够联网的路由器或者交换机上面测试,因为本实验要用到外网。
  2. 本实验串口每秒打印一次从NTP服务器获取的时间。
  3. 国内免费稳定的SNTP服务器很少,当前使用的这个时好时坏,如果大家测试不成功的话也是正常的。

实验操作:

详见本章节30.6小节。

配置向导文件设置(Net_Config.c):

详见本章节30.4小节。

调试文件设置(Net_Debug.c):

详见本章节30.5小节

程序设计:

主函数初始化

在main.c文件实现:

  1. /*
  2.  
  3. *********************************************************************************************************
  4.  
  5. * 函 数 名: main
  6.  
  7. * 功能说明: 标准c程序入口。
  8.  
  9. * 形 参: 无
  10.  
  11. * 返 回 值: 无
  12.  
  13. *********************************************************************************************************
  14.  
  15. */
  16.  
  17. int main (void)
  18.  
  19. {
  20.  
  21. /* 初始化外设 */
  22.  
  23. bsp_Init();
  24.  
  25. /* 进入RL-TCPnet测试函数 */
  26.  
  27. TCPnetTest();
  28.  
  29. }

硬件外设初始化

硬件外设的初始化是在 bsp.c 文件实现:

  1. /*
  2.  
  3. *********************************************************************************************************
  4.  
  5. * 函 数 名: bsp_Init
  6.  
  7. * 功能说明: 初始化所有的硬件设备。该函数配置CPU寄存器和外设的寄存器并初始化一些全局变量。只需要调用一次
  8.  
  9. * 形 参:无
  10.  
  11. * 返 回 值: 无
  12.  
  13. *********************************************************************************************************
  14.  
  15. */
  16.  
  17. void bsp_Init(void)
  18.  
  19. {
  20.  
  21. /*
  22.  
  23. 由于ST固件库的启动文件已经执行了CPU系统时钟的初始化,所以不必再次重复配置系统时钟。
  24.  
  25. 启动文件配置了CPU主时钟频率、内部Flash访问速度和可选的外部SRAM FSMC初始化。
  26.  
  27. 系统时钟缺省配置为168MHz,如果需要更改,可以修改 system_stm32f4xx.c 文件
  28.  
  29. */
  30.  
  31. /* 优先级分组设置为4,可配置0-15级抢占式优先级,0级子优先级,即不存在子优先级。*/
  32.  
  33. NVIC_PriorityGroupConfig(NVIC_PriorityGroup_4);
  34.  
  35. bsp_InitUart(); /* 初始化串口 */
  36.  
  37. bsp_InitKey(); /* 初始化按键变量(必须在 bsp_InitTimer() 之前调用) */
  38.  
  39. bsp_InitLed(); /* 初始LED指示灯端口 */
  40.  
  41. bsp_InitTimer(); /* 初始化滴答定时器 */
  42.  
  43. }

RL-TCPnet功能测试

这里专门创建了一个app_tcpnet_lib.c文件用于RL-TCPnet功能的测试,此文件主要实现从远程NTP服务器获取当前日期和时间,精度到秒。

  1. /*
  2.  
  3. *********************************************************************************************************
  4.  
  5. * 用于本文件的调试
  6.  
  7. *********************************************************************************************************
  8.  
  9. */
  10.  
  11. #if 1
  12.  
  13. #define printf_debug printf
  14.  
  15. #else
  16.  
  17. #define printf_debug(...)
  18.  
  19. #endif
  20.  
  21. /*
  22.  
  23. **********************************************************************************************************
  24.  
  25. 变量
  26.  
  27. **********************************************************************************************************
  28.  
  29. */
  30.  
  31. U8 ntp_server[] = {,,,}; /* SNTP服务器地址 */
  32.  
  33. /*
  34.  
  35. *********************************************************************************************************
  36.  
  37. * 函 数 名: time_cback
  38.  
  39. * 功能说明: SNTP获取时间回到函数
  40.  
  41. * 形 参: 无
  42.  
  43. * 返 回 值: 无
  44.  
  45. *********************************************************************************************************
  46.  
  47. */
  48.  
  49. static void time_cback (uint32_t time)
  50.  
  51. {
  52.  
  53. struct tm *t_tm;
  54.  
  55. if (time == )
  56.  
  57. {
  58.  
  59. printf_debug ("错误, 服务器未响应或者网络状态比较差\r\n");
  60.  
  61. }
  62.  
  63. else
  64.  
  65. {
  66.  
  67. /* 由于是格林尼治时间,所以北京时间要加8个小时 */
  68.  
  69. time += **;
  70.  
  71. /* 将获取的UNIX时间戳转换成年月日时分秒的格式 */
  72.  
  73. t_tm = localtime((unsigned int *)&time);
  74.  
  75. /* 格林尼治时间是从1900年开始的,这要加上 */
  76.  
  77. t_tm->tm_year += ;
  78.  
  79. /* 获取的月份范围是0-11,这里要加1 */
  80.  
  81. t_tm->tm_mon += ;
  82.  
  83. printf_debug ("UNIX时间戳:%d 日期:%02d/%02d/%02d 时间:%02d:%02d:%02d\r\n",
  84.  
  85. time,
  86.  
  87. t_tm->tm_year,
  88.  
  89. t_tm->tm_mon,
  90.  
  91. t_tm->tm_mday,
  92.  
  93. t_tm->tm_hour,
  94.  
  95. t_tm->tm_min,
  96.  
  97. t_tm->tm_sec);
  98.  
  99. }
  100.  
  101. }
  102.  
  103. /*
  104.  
  105. *********************************************************************************************************
  106.  
  107. * 函 数 名: get_time
  108.  
  109. * 功能说明: 从SNTP服务器获取当前时间
  110.  
  111. * 形 参: 无
  112.  
  113. * 返 回 值: 无
  114.  
  115. *********************************************************************************************************
  116.  
  117. */
  118.  
  119. static void get_time (void)
  120.  
  121. {
  122.  
  123. if (sntp_get_time((U8 *)&ntp_server[], &time_cback) == __TRUE)
  124.  
  125. {
  126.  
  127. printf_debug ("SNTP请求已经发送成功\r\n");
  128.  
  129. }
  130.  
  131. else
  132.  
  133. {
  134.  
  135. printf_debug ("失败, SNTP未就绪或者参数错误\r\n");
  136.  
  137. }
  138.  
  139. }
  140.  
  141. /*
  142.  
  143. *********************************************************************************************************
  144.  
  145. * 函 数 名: tcpnet_poll
  146.  
  147. * 功能说明: 使用TCPnet必须要一直调用的函数
  148.  
  149. * 形 参: 无
  150.  
  151. * 返 回 值: 无
  152.  
  153. *********************************************************************************************************
  154.  
  155. */
  156.  
  157. void tcpnet_poll(void)
  158.  
  159. {
  160.  
  161. if(bsp_CheckTimer())
  162.  
  163. {
  164.  
  165. bsp_LedToggle();
  166.  
  167. /* 此函数坚决不可以放在中断里面跑 */
  168.  
  169. timer_tick ();
  170.  
  171. }
  172.  
  173. main_TcpNet ();
  174.  
  175. }
  176.  
  177. /*
  178.  
  179. *********************************************************************************************************
  180.  
  181. * 函 数 名: TCPnetTest
  182.  
  183. * 功能说明: TCPnet应用
  184.  
  185. * 形 参: 无
  186.  
  187. * 返 回 值: 无
  188.  
  189. *********************************************************************************************************
  190.  
  191. */
  192.  
  193. void TCPnetTest(void)
  194.  
  195. {
  196.  
  197. /* 初始化网络协议栈 */
  198.  
  199. init_TcpNet ();
  200.  
  201. /* 创建一个周期是100ms的软定时器 */
  202.  
  203. bsp_StartAutoTimer(, );
  204.  
  205. /* 再创建一个周期是1000ms的软定时器 */
  206.  
  207. bsp_StartAutoTimer(, );
  208.  
  209. while ()
  210.  
  211. {
  212.  
  213. /* TCP轮询 */
  214.  
  215. tcpnet_poll();
  216.  
  217. /* 每秒从NTP服务器获取一次时间 */
  218.  
  219. if(bsp_CheckTimer())
  220.  
  221. {
  222.  
  223. get_time();
  224.  
  225. }
  226.  
  227. }
  228.  
  229. }

30.8 实验例程说明(RTX)

30.8.1 STM32F407开发板实验

配套例子:

V5-1041_RL-TCPnet实验SNTP应用(RTX)

实验目的:

  1. 学习RL-TCPnet的SNTP使用。

实验内容:

  1. 务必将网线接到能够联网的路由器或者交换机上面测试,因为本实验要用到外网。
  2. 本实验串口每秒打印一次从NTP服务器获取的时间。
  3. 国内免费稳定的SNTP服务器很少,当前使用的这个时好时坏,如果大家测试不成功的话也是正常的。

实验操作:

详见本章节30.6小节。

配置向导文件设置(Net_Config.c):

详见本章节30.4小节。

调试文件设置(Net_Debug.c):

详见本章节30.5小节。

RTX配置:

RTX配置向导详情如下:

Task Configuration

(1)Number of concurrent running tasks

允许创建6个任务,实际创建了如下5个任务:

AppTaskUserIF任务   :按键消息处理。

AppTaskLED任务     :LED闪烁。

AppTaskMsgPro任务 :按键检测。

AppTaskTCPMain任务:RL-TCPnet测试任务。

AppTaskStart任务  :启动任务,也是最高优先级任务,这里实现RL-TCPnet的时间基准更新。

(2)Number of tasks with user-provided stack

创建的5个任务都是采用自定义堆栈方式。

(3)Run in privileged mode

设置任务运行在非特权级模式。

RTX任务调试信息:

程序设计:

任务栈大小分配:

static uint64_t AppTaskUserIFStk[1024/8];   /* 任务栈 */

static uint64_t AppTaskLEDStk[1024/8];      /* 任务栈 */

static uint64_t AppTaskMsgProStk[1024/8];  /* 任务栈 */

static uint64_t AppTaskTCPMainStk[2048/8]; /* 任务栈 */

static uint64_t AppTaskStartStk[1024/8];     /* 任务栈 */

将任务栈定义成uint64_t类型可以保证任务栈是8字节对齐的,8字节对齐的含义就是数组的首地址对8求余等于0。如果不做8字节对齐的话,部分C语言库函数、浮点运算和uint64_t类型数据运算会出问题。

系统栈大小分配:

RTX初始化:

  1. /*
  2.  
  3. *********************************************************************************************************
  4.  
  5. * 函 数 名: main
  6.  
  7. * 功能说明: 标准c程序入口。
  8.  
  9. * 形 参: 无
  10.  
  11. * 返 回 值: 无
  12.  
  13. *********************************************************************************************************
  14.  
  15. */
  16.  
  17. int main (void)
  18.  
  19. {
  20.  
  21. /* 初始化外设 */
  22.  
  23. bsp_Init();
  24.  
  25. /* 创建启动任务 */
  26.  
  27. os_sys_init_user (AppTaskStart, /* 任务函数 */
  28.  
  29. , /* 任务优先级 */
  30.  
  31. &AppTaskStartStk, /* 任务栈 */
  32.  
  33. sizeof(AppTaskStartStk)); /* 任务栈大小,单位字节数 */
  34.  
  35. while();
  36.  
  37. }

硬件外设初始化

硬件外设的初始化是在 bsp.c 文件实现:

  1. /*
  2.  
  3. *********************************************************************************************************
  4.  
  5. * 函 数 名: bsp_Init
  6.  
  7. * 功能说明: 初始化所有的硬件设备。该函数配置CPU寄存器和外设的寄存器并初始化一些全局变量。只需要调用一次
  8.  
  9. * 形 参:无
  10.  
  11. * 返 回 值: 无
  12.  
  13. *********************************************************************************************************
  14.  
  15. */
  16.  
  17. void bsp_Init(void)
  18.  
  19. {
  20.  
  21. /*
  22.  
  23. 由于ST固件库的启动文件已经执行了CPU系统时钟的初始化,所以不必再次重复配置系统时钟。
  24.  
  25. 启动文件配置了CPU主时钟频率、内部Flash访问速度和可选的外部SRAM FSMC初始化。
  26.  
  27. 系统时钟缺省配置为168MHz,如果需要更改,可以修改 system_stm32f4xx.c 文件
  28.  
  29. */
  30.  
  31. /* 优先级分组设置为4,可配置0-15级抢占式优先级,0级子优先级,即不存在子优先级。*/
  32.  
  33. NVIC_PriorityGroupConfig(NVIC_PriorityGroup_4);
  34.  
  35. bsp_InitDWT(); /* 初始化DWT */
  36.  
  37. bsp_InitUart(); /* 初始化串口 */
  38.  
  39. bsp_InitKey(); /* 初始化按键变量(必须在 bsp_InitTimer() 之前调用) */
  40.  
  41. bsp_InitLed(); /* 初始LED指示灯端口 */
  42.  
  43. }

RTX任务创建:

  1. /*
  2.  
  3. *********************************************************************************************************
  4.  
  5. * 函 数 名: AppTaskCreate
  6.  
  7. * 功能说明: 创建应用任务
  8.  
  9. * 形 参: 无
  10.  
  11. * 返 回 值: 无
  12.  
  13. *********************************************************************************************************
  14.  
  15. */
  16.  
  17. static void AppTaskCreate (void)
  18.  
  19. {
  20.  
  21. HandleTaskUserIF = os_tsk_create_user(AppTaskUserIF, /* 任务函数 */
  22.  
  23. , /* 任务优先级 */
  24.  
  25. &AppTaskUserIFStk, /* 任务栈 */
  26.  
  27. sizeof(AppTaskUserIFStk)); /* 任务栈大小,单位字节数 */
  28.  
  29. HandleTaskLED = os_tsk_create_user(AppTaskLED, /* 任务函数 */
  30.  
  31. , /* 任务优先级 */
  32.  
  33. &AppTaskLEDStk, /* 任务栈 */
  34.  
  35. sizeof(AppTaskLEDStk)); /* 任务栈大小,单位字节数 */
  36.  
  37. HandleTaskMsgPro = os_tsk_create_user(AppTaskMsgPro, /* 任务函数 */
  38.  
  39. , /* 任务优先级 */
  40.  
  41. &AppTaskMsgProStk, /* 任务栈 */
  42.  
  43. sizeof(AppTaskMsgProStk)); /* 任务栈大小,单位字节数 */
  44.  
  45. HandleTaskTCPMain = os_tsk_create_user(AppTaskTCPMain, /* 任务函数 */
  46.  
  47. , /* 任务优先级 */
  48.  
  49. &AppTaskTCPMainStk, /* 任务栈 */
  50.  
  51. sizeof(AppTaskTCPMainStk)); /* 任务栈大小,单位字节数 */
  52.  
  53. }

五个RTX任务的实现:

  1. /*
  2.  
  3. *********************************************************************************************************
  4.  
  5. * 函 数 名: AppTaskUserIF
  6.  
  7. * 功能说明: 按键消息处理
  8.  
  9. * 形 参: 无
  10.  
  11. * 返 回 值: 无
  12.  
  13. * 优 先 级: 1 (数值越小优先级越低,这个跟uCOS相反)
  14.  
  15. *********************************************************************************************************
  16.  
  17. */
  18.  
  19. __task void AppTaskUserIF(void)
  20.  
  21. {
  22.  
  23. uint8_t ucKeyCode;
  24.  
  25. while()
  26.  
  27. {
  28.  
  29. ucKeyCode = bsp_GetKey();
  30.  
  31. if (ucKeyCode != KEY_NONE)
  32.  
  33. {
  34.  
  35. switch (ucKeyCode)
  36.  
  37. {
  38.  
  39. /* K1键按下 */
  40.  
  41. case KEY_DOWN_K1:
  42.  
  43. printf("K1键按下 \r\n");
  44.  
  45. break;
  46.  
  47. /* K2键按下 */
  48.  
  49. case KEY_DOWN_K2:
  50.  
  51. printf("K2键按下 \r\n");
  52.  
  53. break;
  54.  
  55. /* K3键按下 */
  56.  
  57. case KEY_DOWN_K3:
  58.  
  59. printf("K3键按下 \r\n");
  60.  
  61. break;
  62.  
  63. /* 其他的键值不处理 */
  64.  
  65. default:
  66.  
  67. break;
  68.  
  69. }
  70.  
  71. }
  72.  
  73. os_dly_wait();
  74.  
  75. }
  76.  
  77. }
  78.  
  79. /*
  80.  
  81. *********************************************************************************************************
  82.  
  83. * 函 数 名: AppTaskLED
  84.  
  85. * 功能说明: LED闪烁。
  86.  
  87. * 形 参: 无
  88.  
  89. * 返 回 值: 无
  90.  
  91. * 优 先 级: 2
  92.  
  93. *********************************************************************************************************
  94.  
  95. */
  96.  
  97. __task void AppTaskLED(void)
  98.  
  99. {
  100.  
  101. const uint16_t usFrequency = ; /* 延迟周期 */
  102.  
  103. /* 设置延迟周期 */
  104.  
  105. os_itv_set(usFrequency);
  106.  
  107. while()
  108.  
  109. {
  110.  
  111. bsp_LedToggle();
  112.  
  113. /* os_itv_wait是绝对延迟,os_dly_wait是相对延迟。*/
  114.  
  115. os_itv_wait();
  116.  
  117. }
  118.  
  119. }
  120.  
  121. /*
  122.  
  123. *********************************************************************************************************
  124.  
  125. * 函 数 名: AppTaskMsgPro
  126.  
  127. * 功能说明: 按键检测
  128.  
  129. * 形 参: 无
  130.  
  131. * 返 回 值: 无
  132.  
  133. * 优 先 级: 3
  134.  
  135. *********************************************************************************************************
  136.  
  137. */
  138.  
  139. __task void AppTaskMsgPro(void)
  140.  
  141. {
  142.  
  143. while()
  144.  
  145. {
  146.  
  147. bsp_KeyScan();
  148.  
  149. os_dly_wait();
  150.  
  151. }
  152.  
  153. }
  154.  
  155. /*
  156.  
  157. *********************************************************************************************************
  158.  
  159. * 函 数 名: AppTaskTCPMain
  160.  
  161. * 功能说明: RL-TCPnet测试任务
  162.  
  163. * 形 参: 无
  164.  
  165. * 返 回 值: 无
  166.  
  167. * 优 先 级: 4
  168.  
  169. *********************************************************************************************************
  170.  
  171. */
  172.  
  173. __task void AppTaskTCPMain(void)
  174.  
  175. {
  176.  
  177. while ()
  178.  
  179. {
  180.  
  181. TCPnetTest();
  182.  
  183. }
  184.  
  185. }
  186.  
  187. /*
  188.  
  189. *********************************************************************************************************
  190.  
  191. * 函 数 名: AppTaskStart
  192.  
  193. * 功能说明: 启动任务,也是最高优先级任务,这里实现RL-TCPnet的时间基准更新。
  194.  
  195. * 形 参: 无
  196.  
  197. * 返 回 值: 无
  198.  
  199. * 优 先 级: 5
  200.  
  201. *********************************************************************************************************
  202.  
  203. */
  204.  
  205. __task void AppTaskStart(void)
  206.  
  207. {
  208.  
  209. /* 初始化RL-TCPnet */
  210.  
  211. init_TcpNet ();
  212.  
  213. /* 创建任务 */
  214.  
  215. AppTaskCreate();
  216.  
  217. os_itv_set ();
  218.  
  219. while()
  220.  
  221. {
  222.  
  223. os_itv_wait ();
  224.  
  225. /* RL-TCPnet时间基准更新函数 */
  226.  
  227. timer_tick ();
  228.  
  229. }
  230.  
  231. }

RL-TCPnet功能测试

这里专门创建了一个app_tcpnet_lib.c文件用于RL-TCPnet功能的测试,此文件主要实现从远程NTP服务器获取当前日期和时间,精度到秒。

  1. #include "includes.h"
  2.  
  3. /*
  4.  
  5. *********************************************************************************************************
  6.  
  7. * 用于本文件的调试
  8.  
  9. *********************************************************************************************************
  10.  
  11. */
  12.  
  13. #if 1
  14.  
  15. #define printf_debug printf
  16.  
  17. #else
  18.  
  19. #define printf_debug(...)
  20.  
  21. #endif
  22.  
  23. /*
  24.  
  25. **********************************************************************************************************
  26.  
  27. 变量
  28.  
  29. **********************************************************************************************************
  30.  
  31. */
  32.  
  33. U8 ntp_server[] = {,,,}; /* SNTP服务器地址 */
  34.  
  35. /*
  36.  
  37. *********************************************************************************************************
  38.  
  39. * 函 数 名: time_cback
  40.  
  41. * 功能说明: SNTP获取时间回到函数
  42.  
  43. * 形 参: 无
  44.  
  45. * 返 回 值: 无
  46.  
  47. *********************************************************************************************************
  48.  
  49. */
  50.  
  51. static void time_cback (uint32_t time)
  52.  
  53. {
  54.  
  55. struct tm *t_tm;
  56.  
  57. if (time == )
  58.  
  59. {
  60.  
  61. printf_debug ("错误, 服务器未响应或者网络状态比较差\r\n");
  62.  
  63. }
  64.  
  65. else
  66.  
  67. {
  68.  
  69. /* 由于是格林尼治时间,所以北京时间要加8个小时 */
  70.  
  71. time += **;
  72.  
  73. /* 将获取的UNIX时间戳转换成年月日时分秒的格式 */
  74.  
  75. t_tm = localtime((unsigned int *)&time);
  76.  
  77. /* 格林尼治时间是从1900年开始的,这要加上 */
  78.  
  79. t_tm->tm_year += ;
  80.  
  81. /* 获取的月份范围是0-11,这里要加1 */
  82.  
  83. t_tm->tm_mon += ;
  84.  
  85. printf_debug ("UNIX时间戳:%d 日期:%02d/%02d/%02d 时间:%02d:%02d:%02d\r\n",
  86.  
  87. time,
  88.  
  89. t_tm->tm_year,
  90.  
  91. t_tm->tm_mon,
  92.  
  93. t_tm->tm_mday,
  94.  
  95. t_tm->tm_hour,
  96.  
  97. t_tm->tm_min,
  98.  
  99. t_tm->tm_sec);
  100.  
  101. }
  102.  
  103. }
  104.  
  105. /*
  106.  
  107. *********************************************************************************************************
  108.  
  109. * 函 数 名: get_time
  110.  
  111. * 功能说明: 从SNTP服务器获取当前时间
  112.  
  113. * 形 参: 无
  114.  
  115. * 返 回 值: 无
  116.  
  117. *********************************************************************************************************
  118.  
  119. */
  120.  
  121. static void get_time (void)
  122.  
  123. {
  124.  
  125. if (sntp_get_time((U8 *)&ntp_server[], &time_cback) == __TRUE)
  126.  
  127. {
  128.  
  129. printf_debug ("SNTP请求已经发送成功\r\n");
  130.  
  131. }
  132.  
  133. else
  134.  
  135. {
  136.  
  137. printf_debug ("失败, SNTP未就绪或者参数错误\r\n");
  138.  
  139. }
  140.  
  141. }
  142.  
  143. /*
  144.  
  145. *********************************************************************************************************
  146.  
  147. * 函 数 名: TCPnetTest
  148.  
  149. * 功能说明: TCPnet应用
  150.  
  151. * 形 参: 无
  152.  
  153. * 返 回 值: 无
  154.  
  155. *********************************************************************************************************
  156.  
  157. */
  158.  
  159. void TCPnetTest(void)
  160.  
  161. {
  162.  
  163. uint32_t tstart, tend;
  164.  
  165. /* 初始化起始时间 */
  166.  
  167. tstart = os_time_get();
  168.  
  169. while ()
  170.  
  171. {
  172.  
  173. /* RL-TCPnet处理函数 */
  174.  
  175. main_TcpNet();
  176.  
  177. /* 每秒从NTP服务器获取一次时间 */
  178.  
  179. tend = os_time_get() - tstart;
  180.  
  181. if(tend >= )
  182.  
  183. {
  184.  
  185. tstart = os_time_get();
  186.  
  187. get_time();
  188.  
  189. }
  190.  
  191. os_dly_wait();
  192.  
  193. }
  194.  
  195. }

30.8.2 STM32F429开发板实验

配套例子:

V6-1041_RL-TCPnet实验SNTP应用(RTX)

实验目的:

  1. 学习RL-TCPnet的SNTP使用。

实验内容:

  1. 务必将网线接到能够联网的路由器或者交换机上面测试,因为本实验要用到外网。
  2. 本实验串口每秒打印一次从NTP服务器获取的时间。
  3. 国内免费稳定的SNTP服务器很少,当前使用的这个时好时坏,如果大家测试不成功的话也是正常的。

实验操作:

详见本章节30.6小节。

配置向导文件设置(Net_Config.c):

详见本章节30.4小节。

调试文件设置(Net_Debug.c):

详见本章节30.5小节。

RTX配置:

RTX配置向导详情如下:

Task Configuration

(1)Number of concurrent running tasks

允许创建6个任务,实际创建了如下5个任务:

AppTaskUserIF任务   :按键消息处理。

AppTaskLED任务     :LED闪烁。

AppTaskMsgPro任务 :按键检测。

AppTaskTCPMain任务:RL-TCPnet测试任务。

AppTaskStart任务  :启动任务,也是最高优先级任务,这里实现RL-TCPnet的时间基准更新。

(2)Number of tasks with user-provided stack

创建的5个任务都是采用自定义堆栈方式。

(3)Run in privileged mode

设置任务运行在非特权级模式。

RTX任务调试信息:

程序设计:

任务栈大小分配:

static uint64_t AppTaskUserIFStk[1024/8];   /* 任务栈 */

static uint64_t AppTaskLEDStk[1024/8];      /* 任务栈 */

static uint64_t AppTaskMsgProStk[1024/8];  /* 任务栈 */

static uint64_t AppTaskTCPMainStk[2048/8]; /* 任务栈 */

static uint64_t AppTaskStartStk[1024/8];     /* 任务栈 */

将任务栈定义成uint64_t类型可以保证任务栈是8字节对齐的,8字节对齐的含义就是数组的首地址对8求余等于0。如果不做8字节对齐的话,部分C语言库函数、浮点运算和uint64_t类型数据运算会出问题。

系统栈大小分配:

RTX初始化:

  1. /*
  2.  
  3. *********************************************************************************************************
  4.  
  5. * 函 数 名: main
  6.  
  7. * 功能说明: 标准c程序入口。
  8.  
  9. * 形 参: 无
  10.  
  11. * 返 回 值: 无
  12.  
  13. *********************************************************************************************************
  14.  
  15. */
  16.  
  17. int main (void)
  18.  
  19. {
  20.  
  21. /* 初始化外设 */
  22.  
  23. bsp_Init();
  24.  
  25. /* 创建启动任务 */
  26.  
  27. os_sys_init_user (AppTaskStart, /* 任务函数 */
  28.  
  29. , /* 任务优先级 */
  30.  
  31. &AppTaskStartStk, /* 任务栈 */
  32.  
  33. sizeof(AppTaskStartStk)); /* 任务栈大小,单位字节数 */
  34.  
  35. while();
  36.  
  37. }

硬件外设初始化

硬件外设的初始化是在 bsp.c 文件实现:

  1. /*
  2.  
  3. *********************************************************************************************************
  4.  
  5. * 函 数 名: bsp_Init
  6.  
  7. * 功能说明: 初始化所有的硬件设备。该函数配置CPU寄存器和外设的寄存器并初始化一些全局变量。只需要调用一次
  8.  
  9. * 形 参:无
  10.  
  11. * 返 回 值: 无
  12.  
  13. *********************************************************************************************************
  14.  
  15. */
  16.  
  17. void bsp_Init(void)
  18.  
  19. {
  20.  
  21. /*
  22.  
  23. 由于ST固件库的启动文件已经执行了CPU系统时钟的初始化,所以不必再次重复配置系统时钟。
  24.  
  25. 启动文件配置了CPU主时钟频率、内部Flash访问速度和可选的外部SRAM FSMC初始化。
  26.  
  27. 系统时钟缺省配置为168MHz,如果需要更改,可以修改 system_stm32f4xx.c 文件
  28.  
  29. */
  30.  
  31. /* 优先级分组设置为4,可配置0-15级抢占式优先级,0级子优先级,即不存在子优先级。*/
  32.  
  33. NVIC_PriorityGroupConfig(NVIC_PriorityGroup_4);
  34.  
  35. SystemCoreClockUpdate(); /* 根据PLL配置更新系统时钟频率变量 SystemCoreClock */
  36.  
  37. bsp_InitDWT(); /* 初始化DWT */
  38.  
  39. bsp_InitUart(); /* 初始化串口 */
  40.  
  41. bsp_InitKey(); /* 初始化按键变量(必须在 bsp_InitTimer() 之前调用) */
  42.  
  43. bsp_InitExtIO(); /* FMC总线上扩展了32位输出IO, 操作LED等外设必须初始化 */
  44.  
  45. bsp_InitLed(); /* 初始LED指示灯端口 */
  46.  
  47. }

RTX任务创建:

  1. /*
  2.  
  3. *********************************************************************************************************
  4.  
  5. * 函 数 名: AppTaskCreate
  6.  
  7. * 功能说明: 创建应用任务
  8.  
  9. * 形 参: 无
  10.  
  11. * 返 回 值: 无
  12.  
  13. *********************************************************************************************************
  14.  
  15. */
  16.  
  17. static void AppTaskCreate (void)
  18.  
  19. {
  20.  
  21. HandleTaskUserIF = os_tsk_create_user(AppTaskUserIF, /* 任务函数 */
  22.  
  23. , /* 任务优先级 */
  24.  
  25. &AppTaskUserIFStk, /* 任务栈 */
  26.  
  27. sizeof(AppTaskUserIFStk)); /* 任务栈大小,单位字节数 */
  28.  
  29. HandleTaskLED = os_tsk_create_user(AppTaskLED, /* 任务函数 */
  30.  
  31. , /* 任务优先级 */
  32.  
  33. &AppTaskLEDStk, /* 任务栈 */
  34.  
  35. sizeof(AppTaskLEDStk)); /* 任务栈大小,单位字节数 */
  36.  
  37. HandleTaskMsgPro = os_tsk_create_user(AppTaskMsgPro, /* 任务函数 */
  38.  
  39. , /* 任务优先级 */
  40.  
  41. &AppTaskMsgProStk, /* 任务栈 */
  42.  
  43. sizeof(AppTaskMsgProStk)); /* 任务栈大小,单位字节数 */
  44.  
  45. HandleTaskTCPMain = os_tsk_create_user(AppTaskTCPMain, /* 任务函数 */
  46.  
  47. , /* 任务优先级 */
  48.  
  49. &AppTaskTCPMainStk, /* 任务栈 */
  50.  
  51. sizeof(AppTaskTCPMainStk)); /* 任务栈大小,单位字节数 */
  52.  
  53. }

五个RTX任务的实现:

  1. /*
  2.  
  3. *********************************************************************************************************
  4.  
  5. * 函 数 名: AppTaskUserIF
  6.  
  7. * 功能说明: 按键消息处理
  8.  
  9. * 形 参: 无
  10.  
  11. * 返 回 值: 无
  12.  
  13. * 优 先 级: 1 (数值越小优先级越低,这个跟uCOS相反)
  14.  
  15. *********************************************************************************************************
  16.  
  17. */
  18.  
  19. __task void AppTaskUserIF(void)
  20.  
  21. {
  22.  
  23. uint8_t ucKeyCode;
  24.  
  25. while()
  26.  
  27. {
  28.  
  29. ucKeyCode = bsp_GetKey();
  30.  
  31. if (ucKeyCode != KEY_NONE)
  32.  
  33. {
  34.  
  35. switch (ucKeyCode)
  36.  
  37. {
  38.  
  39. /* K1键按下 */
  40.  
  41. case KEY_DOWN_K1:
  42.  
  43. printf("K1键按下 \r\n");
  44.  
  45. break;
  46.  
  47. /* K2键按下 */
  48.  
  49. case KEY_DOWN_K2:
  50.  
  51. printf("K2键按下 \r\n");
  52.  
  53. break;
  54.  
  55. /* K3键按下 */
  56.  
  57. case KEY_DOWN_K3:
  58.  
  59. printf("K3键按下 \r\n");
  60.  
  61. break;
  62.  
  63. /* 其他的键值不处理 */
  64.  
  65. default:
  66.  
  67. break;
  68.  
  69. }
  70.  
  71. }
  72.  
  73. os_dly_wait();
  74.  
  75. }
  76.  
  77. }
  78.  
  79. /*
  80.  
  81. *********************************************************************************************************
  82.  
  83. * 函 数 名: AppTaskLED
  84.  
  85. * 功能说明: LED闪烁。
  86.  
  87. * 形 参: 无
  88.  
  89. * 返 回 值: 无
  90.  
  91. * 优 先 级: 2
  92.  
  93. *********************************************************************************************************
  94.  
  95. */
  96.  
  97. __task void AppTaskLED(void)
  98.  
  99. {
  100.  
  101. const uint16_t usFrequency = ; /* 延迟周期 */
  102.  
  103. /* 设置延迟周期 */
  104.  
  105. os_itv_set(usFrequency);
  106.  
  107. while()
  108.  
  109. {
  110.  
  111. bsp_LedToggle();
  112.  
  113. /* os_itv_wait是绝对延迟,os_dly_wait是相对延迟。*/
  114.  
  115. os_itv_wait();
  116.  
  117. }
  118.  
  119. }
  120.  
  121. /*
  122.  
  123. *********************************************************************************************************
  124.  
  125. * 函 数 名: AppTaskMsgPro
  126.  
  127. * 功能说明: 按键检测
  128.  
  129. * 形 参: 无
  130.  
  131. * 返 回 值: 无
  132.  
  133. * 优 先 级: 3
  134.  
  135. *********************************************************************************************************
  136.  
  137. */
  138.  
  139. __task void AppTaskMsgPro(void)
  140.  
  141. {
  142.  
  143. while()
  144.  
  145. {
  146.  
  147. bsp_KeyScan();
  148.  
  149. os_dly_wait();
  150.  
  151. }
  152.  
  153. }
  154.  
  155. /*
  156.  
  157. *********************************************************************************************************
  158.  
  159. * 函 数 名: AppTaskTCPMain
  160.  
  161. * 功能说明: RL-TCPnet测试任务
  162.  
  163. * 形 参: 无
  164.  
  165. * 返 回 值: 无
  166.  
  167. * 优 先 级: 4
  168.  
  169. *********************************************************************************************************
  170.  
  171. */
  172.  
  173. __task void AppTaskTCPMain(void)
  174.  
  175. {
  176.  
  177. while ()
  178.  
  179. {
  180.  
  181. TCPnetTest();
  182.  
  183. }
  184.  
  185. }
  186.  
  187. /*
  188.  
  189. *********************************************************************************************************
  190.  
  191. * 函 数 名: AppTaskStart
  192.  
  193. * 功能说明: 启动任务,也是最高优先级任务,这里实现RL-TCPnet的时间基准更新。
  194.  
  195. * 形 参: 无
  196.  
  197. * 返 回 值: 无
  198.  
  199. * 优 先 级: 5
  200.  
  201. *********************************************************************************************************
  202.  
  203. */
  204.  
  205. __task void AppTaskStart(void)
  206.  
  207. {
  208.  
  209. /* 初始化RL-TCPnet */
  210.  
  211. init_TcpNet ();
  212.  
  213. /* 创建任务 */
  214.  
  215. AppTaskCreate();
  216.  
  217. os_itv_set ();
  218.  
  219. while()
  220.  
  221. {
  222.  
  223. os_itv_wait ();
  224.  
  225. /* RL-TCPnet时间基准更新函数 */
  226.  
  227. timer_tick ();
  228.  
  229. }
  230.  
  231. }

RL-TCPnet功能测试

这里专门创建了一个app_tcpnet_lib.c文件用于RL-TCPnet功能的测试,此文件主要实现从远程NTP服务器获取当前日期和时间,精度到秒。

  1. #include "includes.h"
  2.  
  3. /*
  4.  
  5. *********************************************************************************************************
  6.  
  7. * 用于本文件的调试
  8.  
  9. *********************************************************************************************************
  10.  
  11. */
  12.  
  13. #if 1
  14.  
  15. #define printf_debug printf
  16.  
  17. #else
  18.  
  19. #define printf_debug(...)
  20.  
  21. #endif
  22.  
  23. /*
  24.  
  25. **********************************************************************************************************
  26.  
  27. 变量
  28.  
  29. **********************************************************************************************************
  30.  
  31. */
  32.  
  33. U8 ntp_server[] = {,,,}; /* SNTP服务器地址 */
  34.  
  35. /*
  36.  
  37. *********************************************************************************************************
  38.  
  39. * 函 数 名: time_cback
  40.  
  41. * 功能说明: SNTP获取时间回到函数
  42.  
  43. * 形 参: 无
  44.  
  45. * 返 回 值: 无
  46.  
  47. *********************************************************************************************************
  48.  
  49. */
  50.  
  51. static void time_cback (uint32_t time)
  52.  
  53. {
  54.  
  55. struct tm *t_tm;
  56.  
  57. if (time == )
  58.  
  59. {
  60.  
  61. printf_debug ("错误, 服务器未响应或者网络状态比较差\r\n");
  62.  
  63. }
  64.  
  65. else
  66.  
  67. {
  68.  
  69. /* 由于是格林尼治时间,所以北京时间要加8个小时 */
  70.  
  71. time += **;
  72.  
  73. /* 将获取的UNIX时间戳转换成年月日时分秒的格式 */
  74.  
  75. t_tm = localtime((unsigned int *)&time);
  76.  
  77. /* 格林尼治时间是从1900年开始的,这要加上 */
  78.  
  79. t_tm->tm_year += ;
  80.  
  81. /* 获取的月份范围是0-11,这里要加1 */
  82.  
  83. t_tm->tm_mon += ;
  84.  
  85. printf_debug ("UNIX时间戳:%d 日期:%02d/%02d/%02d 时间:%02d:%02d:%02d\r\n",
  86.  
  87. time,
  88.  
  89. t_tm->tm_year,
  90.  
  91. t_tm->tm_mon,
  92.  
  93. t_tm->tm_mday,
  94.  
  95. t_tm->tm_hour,
  96.  
  97. t_tm->tm_min,
  98.  
  99. t_tm->tm_sec);
  100.  
  101. }
  102.  
  103. }
  104.  
  105. /*
  106.  
  107. *********************************************************************************************************
  108.  
  109. * 函 数 名: get_time
  110.  
  111. * 功能说明: 从SNTP服务器获取当前时间
  112.  
  113. * 形 参: 无
  114.  
  115. * 返 回 值: 无
  116.  
  117. *********************************************************************************************************
  118.  
  119. */
  120.  
  121. static void get_time (void)
  122.  
  123. {
  124.  
  125. if (sntp_get_time((U8 *)&ntp_server[], &time_cback) == __TRUE)
  126.  
  127. {
  128.  
  129. printf_debug ("SNTP请求已经发送成功\r\n");
  130.  
  131. }
  132.  
  133. else
  134.  
  135. {
  136.  
  137. printf_debug ("失败, SNTP未就绪或者参数错误\r\n");
  138.  
  139. }
  140.  
  141. }
  142.  
  143. /*
  144.  
  145. *********************************************************************************************************
  146.  
  147. * 函 数 名: TCPnetTest
  148.  
  149. * 功能说明: TCPnet应用
  150.  
  151. * 形 参: 无
  152.  
  153. * 返 回 值: 无
  154.  
  155. *********************************************************************************************************
  156.  
  157. */
  158.  
  159. void TCPnetTest(void)
  160.  
  161. {
  162.  
  163. uint32_t tstart, tend;
  164.  
  165. /* 初始化起始时间 */
  166.  
  167. tstart = os_time_get();
  168.  
  169. while ()
  170.  
  171. {
  172.  
  173. /* RL-TCPnet处理函数 */
  174.  
  175. main_TcpNet();
  176.  
  177. /* 每秒从NTP服务器获取一次时间 */
  178.  
  179. tend = os_time_get() - tstart;
  180.  
  181. if(tend >= )
  182.  
  183. {
  184.  
  185. tstart = os_time_get();
  186.  
  187. get_time();
  188.  
  189. }
  190.  
  191. os_dly_wait();
  192.  
  193. }
  194.  
  195. }

30.9 总结

本章节就为大家讲解这么多,本章节的内容相对比较简单,主要是函数sntp_get_time的使用,希望大家熟练掌握。

【RL-TCPnet网络教程】第30章 RL-TCPnet之SNTP网络时间获取的更多相关文章

  1. Linux网络服务第六章PXE高效能批量网络装机

    1.IP地址配置 2.关闭防火墙以及selinux状态如下 systemctl  stop     firewalld Iptables -F Setenforce 0 三.部署FTP服务 1.安装F ...

  2. 【RL-TCPnet网络教程】第32章 RL-TCPnet之Telnet服务器

    第32章      RL-TCPnet之Telnet服务器 本章节为大家讲解RL-TCPnet的Telnet应用,学习本章节前,务必要优先学习第31章的Telnet基础知识.有了这些基础知识之后,再搞 ...

  3. 【RL-TCPnet网络教程】第28章 RL-TCPnet之DNS应用

    第28章      RL-TCPnet之DNS应用 本章节为大家讲解RL-TCPnet的DNS应用,学习本章节前,务必要优先学习第27章的DNS基础知识.有了这些基础知识之后,再搞本章节会有事半功倍的 ...

  4. 【RL-TCPnet网络教程】第26章 RL-TCPnet之DHCP应用

    第26章     RL-TCPnet之DHCP应用 本章节为大家讲解RL-TCPnet的DHCP应用,学习本章节前,务必要优先学习第25章的DHCP基础知识.有了这些基础知识之后,再搞本章节会有事半功 ...

  5. 【RL-TCPnet网络教程】第20章 RL-TCPnet之BSD Socket客户端

    第20章      RL-TCPnet之BSD Socket客户端 本章节为大家讲解RL-TCPnet的BSD Socket,学习本章节前,务必要优先学习第18章的Socket基础知识.有了这些基础知 ...

  6. 【RL-TCPnet网络教程】第19章 RL-TCPnet之BSD Socket服务器

    第19章      RL-TCPnet之BSD Socket服务器 本章节为大家讲解RL-TCPnet的BSD Socket,学习本章节前,务必要优先学习第18章的Socket基础知识.有了这些基础知 ...

  7. 【RL-TCPnet网络教程】第17章 RL-TCPnet之UDP通信

    第17章      RL-TCPnet之UDP通信 本章节为大家讲解RL-TCPnet的UDP通信实现,学习本章节前,务必要优先学习第16章UDP用户数据报协议基础知识.有了这些基础知识之后,再搞本章 ...

  8. 【RL-TCPnet网络教程】第14章 RL-TCPnet之TCP客户端

    第14章      RL-TCPnet之TCP客户端 本章节为大家讲解RL-TCPnet的TCP客户端实现,学习本章节前,务必要优先学习第12章TCP传输控制协议基础知识.有了这些基础知识之后,再搞本 ...

  9. 【RL-TCPnet网络教程】第13章 RL-TCPnet之TCP服务器

    第13章      RL-TCPnet之TCP服务器 本章节为大家讲解RL-TCPnet的TCP服务器实现,学习本章节前,务必要优先学习第12章TCP传输控制协议基础知识.有了这些基础知识之后,再搞本 ...

随机推荐

  1. JSONP方法简单封装

    function myJsonp(url, data, callback) { //参数判断 if (Object.prototype.toString.call(data) == "[ob ...

  2. YARN配置

    环境搭建 mapred-site.xml <configuration> <property> <name>mapreduce.framework.name< ...

  3. appium 与 selenium python解决python 'WebElement' object does not support indexing 报错问题问题

    再用selenium编写测试脚本时,发现出现python 'WebElement' object does not support indexing 报错问题问题,再找一些解决方法时,发现Appium ...

  4. Bootstrap-datepicker3官方文档中文翻译---I18N/国际化(原文链接 http://bootstrap-datepicker.readthedocs.io/en/latest/index.html)

    I18N/国际化 这个插件支持月份和星期名以及weekStart选项的国际化.默认是英语(“en”); 其他有效的译本语言在 js/locales/ 目录中, 只需在插件后包含您想要的地区. 想要添加 ...

  5. springboot整合mybatis开发

    1创建项目,在启动类上添加映射扫描注解 2导入依赖,添加mybatis generator自动生成代码插件 <!-- mybatis generator 自动生成代码插件 --> < ...

  6. .net基础学java系列(八)SpringBoot

    嘟嘟独立博客 Spring-Boot干货系列 http://tengj.top/categories/Spring-Boot干货系列/ 龙码精神 Java Spring Boot VS .NetCor ...

  7. Mybatis Mapper接口是如何找到实现类的-源码分析

    KeyWords: Mybatis 原理,源码,Mybatis Mapper 接口实现类,代理模式,动态代理,Java动态代理,Proxy.newProxyInstance,Mapper 映射,Map ...

  8. 浅析布隆过滤器及实现demo

    布隆过滤器 布隆过滤器(Bloom Filter)是一种概率空间高效的数据结构.它与hashmap非常相似,用于检索一个元素是否在一个集合中.它在检索元素是否存在时,能很好地取舍空间使用率与误报比例. ...

  9. 在SSL / https下托管SignalR

    https://weblog.west-wind.com/posts/2013/Sep/23/Hosting-SignalR-under-SSLhttps  2013年9月23日•来自毛伊岛,HI• ...

  10. Linux下system()函数的实现

    #include <stdio.h> #include <stdlib.h> #include <unistd.h> #include <sys/types. ...