NTP协议实现

article/2025/10/6 17:45:04
本文出处:
http://www.eefocus.com/html/10-04/94667s.shtml
10.4 实验内容 ——NTP协议实现
1.实验目的
通过实现NTP协议的练习,进一步掌握Linux网络编程,并且提高协议的分析与实现能力,为参与完成综合性项目打下良好的基础。
2.实验内容

     Network Time ProtocolNTP)协议是用来使计算机时间同步化 的一种协议,它可以使计算机对其服务器或时钟源(如石英钟,GPS等)做同步化,它可以提供高精确度的时间校正(LAN上与标准时间差小于1毫秒,WAN上几十毫秒),且可用加密确认的 方式来防止协议攻击。

NTP提供准确时间,首先要有准确的时间来源,这一时间应该是国际标准时间UTC。 NTP获得UTC的时间来源可以是原子钟、天文台、卫星,也可以从Internet上获取。这样就有了准确而可靠的时间源。时间是按NTP服务器的等级传 播。按照距离外部UTC 源的远近将所有服务器归入不同的Stratun(层)中。Stratum-1在顶层,有外部UTC接入,而Stratum-2则从Stratum-1获取 时间,Stratum-3从Stratum-2获取时间,以此类推,但Stratum层的总数限制在15以内。所有这些服务器在逻辑上形成阶梯式的架构并 相互连接,而Stratum-1的时间服务器是整个系统的基础。
进行网络协议实现时最重要的是了解协议数据格式。NTP数据包有48个字节,其中NTP包头16字节,时间戳32个字节。其协议格式如图10.9所示。

图10.9 NTP协议数据格式
其协议字段的含义如下所示。
LI:跳跃指示器,警告在当月最后一天的最终时刻插入的迫近闺秒(闺秒)。
VN:版本号。
Mode:工作模式。该字段包括以下值:0-预留;1-对称行为;3-客户机;4-服务器;5-广播;6-NTP控制信息。NTP协议具有3种工作模式,分别为主/被动对称模式、客户/服务器模式、广播模式。在主/被动对称模式中,有一对一的连接,双方均可同步对方或被对方同步,先发出申请建立连接的一方工作在主动模式下,另一方工作在被动模式下; 客户/服务器模 式与主/被动模式基本相同,惟一区别在于客户方可被服务器同步,但服务器不能被客户同步;在广播模式中,有一对多的连接,服务器不论客户工作 在何种模式下,都会主动发出时间信息,客户根据此信息调整自己的时间。
Stratum:对本地时钟级别的整体识别。
Poll:有符号整数表示连续信息间的最大间隔。
Precision:有符号整数表示本地时钟精确度。
Root Delay:表示到达主参考源的一次往复的总延迟,它是有15~16位小数部分的符号定点小 数。
Root Dispersion:表示一次到达主参考源的标准误差,它是有15~16位小数部分的无符号 定点小数。
Reference Identifier:识别特殊参考源。
Originate Timestamp:这是向服务器请求分离客户机的时间,采用64位时标格式。
Receive Timestamp:这是向服务器请求到达客户机的时间,采用64位时标格式。
Transmit Timestamp:这是向客户机答复分离服务器的时间,采用64位时标格式。
Authenticator(Optional):当实现了NTP认证模式时,主要标识符和信息数字域就 包括已定义的信息认证代码(MAC)信息。
由于NTP协议中涉及比较多的时间相关的操作,为了简化实现过程,在本实验中,仅要求实现NTP协议客户端部分的网络通信模块,也就是构造NTP协议字段 进行发送和接收,最后与时间相关的操作不需进行处理。NTP协议是作为OSI参考模型的高层协议比较适合采用UDP传输协议进行数据传输,专用端口号为 123。在实验中,以国家授时中心服务器(IP地址为 202.72.145.44)作为NTP(网络时间)服务器。 
3.实验步骤
(1)画出流程图。
简易NTP客户端的实现流程如图10.10所示。

图10.10 简易NTP客户端流程图

(2)编写程序。
具体代码如下:
/* ntp.c */
#include <sys/socket.h>
#include <sys/wait.h>
#include <stdio.h>
#include <stdlib.h>
#include <errno.h>
#include <string.h>
#include <sys/un.h>
#include <sys/time.h>
#include <sys/ioctl.h>
#include <unistd.h>
#include <netinet/in.h>
#include <string.h>
#include <netdb.h>
#define NTP_PORT 123               /*NTP专 用端口号字符串*/
#define TIME_PORT 37               /* TIME/UDP端 口号 */
#define NTP_SERVER_IP "210.72.145.44" /*国家授时中心 IP*/
#define NTP_PORT_STR "123"          /*NTP专用端口号字 符串*/
#define NTPV1 "NTP/V1"      /*协议及其版本号*/
#define NTPV2 "NTP/V2"
#define NTPV3 "NTP/V3"
#define NTPV4 "NTP/V4"
#define TIME "TIME/UDP"
#define NTP_PCK_LEN 48
#define LI 0
#define VN 3
#define MODE 3
#define STRATUM 0
#define POLL 4
#define PREC -6
#define JAN_1970 0x83aa7e80 /* 1900年~1970年之间的时间秒数 */
#define NTPFRAC(x) (4294 * (x) + ((1981 * (x)) >> 11))

#define USEC(x)         (((x) >> 12) - 759 * ((((x) >> 10) + 32768) >> 16))

typedef struct _ntp_time
{

    unsigned int coarse;

    unsigned int fine;

} ntp_time;
 
struct ntp_packet
{

     unsigned char leap_ver_mode;

     unsigned char startum;

     char poll;

     char precision;

     int root_delay;
     int root_dispersion;

     int reference_identifier;

     ntp_time reference_timestamp;

     ntp_time originage_timestamp;

     ntp_time receive_timestamp;

     ntp_time transmit_timestamp;

};
 
char protocol[32];
/*构建NTP协议包*/
int construct_packet(char *packet)
{

     char version = 1;

     long tmp_wrd;

int port;
time_t timer;
strcpy(protocol, NTPV3);
/*判断协议版本*/

     if(!strcmp(protocol, NTPV1)||!strcmp(protocol, NTPV2)

           ||!strcmp(protocol, NTPV3)||!strcmp(protocol, NTPV4))

{
memset(packet, 0, NTP_PCK_LEN);
port = NTP_PORT;
/*设置16字节的包头*/
version = protocol[6] - 0x30;

          tmp_wrd = htonl((LI << 30)|(version << 27)

                |(MODE << 24)|(STRATUM << 16)|(POLL << 8)|(PREC & 0xff));

          memcpy(packet, &tmp_wrd, sizeof(tmp_wrd));

/*设置Root Delay、Root Dispersion和Reference Indentifier */

          tmp_wrd = htonl(1<<16);

          memcpy(&packet[4], &tmp_wrd, sizeof(tmp_wrd));

          memcpy(&packet[8], &tmp_wrd, sizeof(tmp_wrd));

/*设置Timestamp部分*/
time(&timer);
/*设置Transmit Timestamp coarse*/

          tmp_wrd = htonl(JAN_1970 + (long)timer);

          memcpy(&packet[40], &tmp_wrd, sizeof(tmp_wrd));

/*设置Transmit Timestamp fine*/

          tmp_wrd = htonl((long)NTPFRAC(timer));

          memcpy(&packet[44], &tmp_wrd, sizeof(tmp_wrd));

          return NTP_PCK_LEN;

}

     else if (!strcmp(protocol, TIME))/* "TIME/UDP" */

{
port = TIME_PORT;
memset(packet, 0, 4);
return 4;
}
return 0;
}
/*获取NTP时间*/

int get_ntp_time(int sk, struct addrinfo *addr, struct ntp_packet *ret_time)

{
fd_set pending_data;

     struct timeval block_time;

     char data[NTP_PCK_LEN * 8];

     int packet_len, data_len = addr->ai_addrlen, count = 0, result, i, re;

     if (!(packet_len = construct_packet(data)))

{

          return 0;

}
/*客户端给服务器端发送NTP协议数据包*/

     if ((result = sendto(sk, data,

             packet_len, 0, addr->ai_addr, data_len)) < 0)

{
perror("sendto");

          return 0;

}
/*调用select()函数,并设定超时时间为1s*/
 FD_ZERO(&pending_data);

     FD_SET(sk, &pending_data);

     block_time.tv_sec=10;
     block_time.tv_usec=0;

     if (select(sk + 1, &pending_data, NULL, NULL, &block_time) > 0)

{
/*接收服务器端的信息*/

          if ((count = recvfrom(sk, data,

                        NTP_PCK_LEN * 8, 0, addr->ai_addr, &data_len)) < 0)

{

               perror("recvfrom");

               return 0;

}

          if (protocol == TIME)

{

               memcpy(&ret_time->transmit_timestamp, data, 4);

               return 1;

}

          else if (count < NTP_PCK_LEN)

{

               return 0;

}
/* 设置接收NTP包的数据结构 */

          ret_time->leap_ver_mode = ntohl(data[0]);

          ret_time->startum = ntohl(data[1]);

          ret_time->poll = ntohl(data[2]);

          ret_time->precision = ntohl(data[3]);

          ret_time->root_delay = ntohl(*(int*)&(data[4]));

          ret_time->root_dispersion = ntohl(*(int*)&(data[8]));

         ret_time->reference_identifier = ntohl(*(int*)&(data[12]));

          ret_time->reference_timestamp.coarse = ntohl *(int*)&(data[16]));

          ret_time->reference_timestamp.fine = ntohl(*(int*)&(data[20]));

          ret_time->originage_timestamp.coarse = ntohl(*(int*)&(data[24]));

          ret_time->originage_timestamp.fine = ntohl(*(int*)&(data[28]));

          ret_time->receive_timestamp.coarse = ntohl(*(int*)&(data[32]));

          ret_time->receive_timestamp.fine = ntohl(*(int*)&(data[36]));

          ret_time->transmit_timestamp.coarse = ntohl(*(int*)&(data[40]));

          ret_time->transmit_timestamp.fine = ntohl(*(int*)&(data[44]));

          return 1;

     } /* end of if select */

     return 0;

}
/* 修改本地时间 */

int set_local_time(struct ntp_packet * pnew_time_packet)

{

     struct timeval tv;

     tv.tv_sec = pnew_time_packet->transmit_timestamp.coarse - JAN_1970;

     tv.tv_usec = USEC(pnew_time_packet->transmit_timestamp.fine);

     return settimeofday(&tv, NULL);

}
int main()
{

     int sockfd, rc;

     struct addrinfo hints, *res = NULL;

     struct ntp_packet new_time_packet;

     memset(&hints, 0, sizeof(hints));

 hints.ai_family = AF_UNSPEC;

     hints.ai_socktype = SOCK_DGRAM;

     hints.ai_protocol = IPPROTO_UDP;
/*调用getaddrinfo()函数,获取地址信息*/

     rc = getaddrinfo(NTP_SERVER_IP, NTP_PORT_STR, &hints, &res);

if (rc != 0)
{
perror("getaddrinfo");
return 1;
}
/* 创建套接字 */

     sockfd = socket(res->ai_family, res->ai_socktype, res->ai_protocol);

if (sockfd <0 )
{
perror("socket");
return 1;

     }         

/*调用取得NTP时间的函数*/

     if (get_ntp_time(sockfd, res, &new_time_packet))

{
/*调整本地时间*/
if (!set_local_time(&new_time_packet))
{

               printf("NTP client success!\n");

}
}    
close(sockfd);

     return 0;

}
为了更好地观察程序的效果,先用date命令修改一下系统时间,再运行实例程序。运行完了之后再查看系统时间,可以发现已经恢复准确的系统时间了。具体运 行结果如下所示。
$ date -s "2001-01-01 1:00:00"
2001年 01月 01日 星期一 01:00:00 EST
$ date
2001年 01月 01日 星期一 01:00:00 EST
$ ./ ntp
NTP client success!
$ date
能够显示当前准确的日期和时间了!

http://chatgpt.dhexx.cn/article/ePJWDhfk.shtml

相关文章

NTP

NTP(Network Time Protocol) 网络时间协议 NTP是网络时间协议(Network Time Protocol)&#xff0c;它是用来同步网络中各个计算机的时间的协议。它的用途是把计算机的时钟同步到世界协调时UTC&#xff0c;其精度在局域网内可达0.1ms&#xff0c;在互联网上绝大多数的地方其精度…

NTP与PTP

NTP&#xff08;Network Time Protocol&#xff0c;网络时间协议&#xff09;是由RFC 1305定义的时间同步协议&#xff0c;用来在分布式时间服务器和客户端之间进行时间同步。NTP基于UDP报文进行传输&#xff0c;使用的UDP端口号为123。 使用NTP的目的是对网络内所有具有时钟的…

通信协议 - ntp时间同步

概述 NTP(Network Time Protocol&#xff09;网络时间协议基于UDP&#xff0c;用于网络时间同步的协议&#xff0c;使网络中的计算机时钟同步到UTC&#xff0c;再配合各个时区的偏移调整就能实现精准同步对时功能。提供NTP对时的服务器有很多&#xff0c;比如微软的NTP对时服务…

NTP协议原理

什么是NTP协议 当第一次听到NTP时&#xff0c;完全不知道是什么概念&#xff0c;只好问度娘。度娘搜到的答案如下&#xff1a; 看了度娘的解释后&#xff0c;才大概明白该协议是进行网络对时的协议&#xff0c;比如当自己的电脑时间不对&#xff0c;可以从另外一台电脑病基于…

网络时间协议NTP

网络时间协议NTP&#xff08;Network Time Protocol&#xff09;是TCP/IP协议族里面的一个应用层协议。NTP用于在一系列分布式时间服务器与客户端之间同步时钟。 随着网络拓扑的日益复杂&#xff0c;整个网络内设备的时钟同步将变得十分重要。如果依靠管理员手工修改系统时钟&a…

NTP网络时间协议

转自https://www.sohu.com/a/206862105_130560 SNPT 秒级时间精度 NTP授时精度与NTP服务器与用户间的网络状况有关&#xff1a;广域网授时精度通常能达50ms级&#xff0c;但有时超过500ms&#xff1b;局域网授时不存在路由器路径延迟问题&#xff0c;因而授时精度理论上可以提…

NTP协议解析

NTP&#xff08;Network Time Protocol&#xff0c;网络时间协议&#xff09;是由RFC 1305定义的时间同步协议&#xff0c;用来在分布式时间服务器和客户端之间进行时间同步。NTP基于UDP报文进行传输&#xff0c;使用的UDP端口号为123。 使用NTP的目的是对网络内所有具有时钟的…

NTP协议详解

前言 NTP(Network Time Protocol&#xff09;网络时间协议基于UDP&#xff0c;用于网络时间同步的协议&#xff0c;使网络中的计算机时钟同步到UTC&#xff0c;再配合各个时区的偏移调整就能实现精准同步对时功能。提供NTP对时的服务器有很多&#xff0c;比如微软的NTP对时服务…

NTP 协议

NTP: Network Time Protocal 一、定义&#xff1a;为实现高精确度的时间同步&#xff0c;而设计的网络时钟同步协议。在Linux系统中&#xff0c;它的最新实现是NTP 4.0&#xff08;一个分布式的网络时钟同步程序&#xff09;。相关定义和实现参看RFC1305和www.ntp.org。NTP协议…

NTP(Network Time Protocol)协议详解

一、NTP的基本概念&#xff1a; NTP(Network Time Protocol)------网络时间协议-----应用层协议&#xff0c;用来在分布式时间服务器和客户端之间进行时间同步。 二、采用NTP的目的&#xff1a; 是对网络内所有具有时钟的设备进行时钟同步&#xff0c;使网络内所有设备的时钟…

Windows10 debug安装、下载

目录 背景下载安装和使用 背景 由于Windows 10不支持debug命令&#xff0c;如果装虚拟机Windows XP就比较耗内存&#xff0c;所以我用的是DOSBox&#xff0c;然后挂载debug程序文件目录。 下载 DOSBox、debug、masm等等&#xff0c;我已经整理好了&#xff0c;下载地址&…

【WINDOWS系统】WinDbg调试工具

一、Windows 10 调试工具 (WinDbg)的安装 1、下载地址https://docs.microsoft.com/zh-cn/windows-hardware/drivers/debugger/debugger-download-tools#small-classic-windbg-preview-logo-debugging-tools-for-windows-10-windbg 2、安装 记录WinDbg安装目录&#xff0c;待会…

windbg 调试崩溃

前言 windbg 是非常强大的调试工具&#xff0c;但是在使用windbg 进修调试时候&#xff0c;很多的命令不知道如何使用。文章简单介绍如何使用windbg进行调试 https://docs.microsoft.com/zh-cn/windows-hardware/drivers/debugger/debugger-download-tools 一、Windows 调试入…

windbg 调试学习

符号服务器地址 微软的公共符号服务器地址&#xff1a;Symbol information 五种符号加载状态 1. deferred: 延迟加载&#xff0c;现在还不需要使用这个模块的符号&#xff0c;将在后面需要的时候加载 2. no symbols: 没有找到相应的符号文件 3. export: 没有找到symbols,但…

Windbg调试(使用方法)

一、Windbg版本信息 Windbg分32位和64位版本&#xff0c;32位程序应使用32位Windbg调试&#xff0c;64位程序应64位Windbg调试。 若想使用64位的Windbg分析32位的程序, 使用如下命令进行CPU模式的切换&#xff1a; .load wow64exts;!sw//例如&#xff1a; 1.查看线程停在哪里…

Windebug

1、 安装drwtsn32&#xff08;Dr.Watson win7没有了&#xff09; 用户可以通过drwtsn32命令&#xff0c;查看dmp文件会被保存在何处。 2、 安装windbg,Windbg下载地址&#xff1a; http://www.microsoft.com/whdc/devtools/debugging/default.mspx 3、 设置windbg A、符号表路径…

windbg调试分析dump工具,使用windbg分析Qt崩溃原因

目录 windbg下载和安装windbg调试exe可执行文件windbg分析dump文件 在非开发环境&#xff0c;windbg工具可以对.exe可执行文件进行调试&#xff0c;也可以分析dump文件&#xff0c;是release版本运行模式解决bug的神器。 windbg下载和安装 一般在安装windows系统软件时会安装w…

02-windows调试工具(DebugDiag使用)

工具下载地址&#xff1a;https://www.microsoft.com/en-us/download/details.aspx?id49924csdn的下载地址&#xff1a;https://download.csdn.net/download/qq_37103755/87089384DebugDiag的简介&#xff1a; DebugDiag 工具主要用于帮助解决如挂起、 速度慢、 内存泄漏或内存…

如何使用Debugging Tools for Windows (windebug)简单的使用心得

1.安装debug工具 ​ 下载页面地址&#xff1a;http://www.microsoft.com/whdc/devtools/debugging/installx86.mspx 选择合适的版本安装 ​ 2.安装Symbols&#xff08;特征库&#xff09; 建议可以多安装以免出现分析不出来的情况 ​ 下载地址&#xff1a;http://www.mi…

WinDebug快速使用及调试注意事项

参考文献&#xff1a; http://www.cnblogs.com/killmyday/archive/2010/03/14/1685331.html WinDbg是微软发布的一款强大的源码级调试工具&#xff0c;支持用户态和内核态两种模式的调试&#xff0c;支持Dump文件分析、内存泄露诊断、同时支持跨机器远程附加进程调试。在具体…