简单理解NTP v3协议

前段时间瞎折腾,给自己的黑莓Bold 9900写了个通过NTP同步时间的小工具,顺便在这里记录一下我在实现一个NTP客户端时对这个协议的理解。

端口号

NTP协议使用UDP作为传输层协议,服务器监听UDP端口123,在收到有效的报文后,服务器会发送响应报文,否则服务器将直接忽略不做响应。

时间格式

NTP协议使用三种时间格式。

NTP短时间格式

短时间格式长度为32位,其中高16位代表从NTP时间戳0秒至现在的秒数,低16位代表1秒以内的分数部分。
这个格式只会在NTP报文的delay和dispersion字段中用到。

1
2
3
4
5
 0                   1                   2                   3
0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5 6 7 8 9 0 1
+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
| Seconds | Fraction |
+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+

NTP时间戳

NTP时间戳格式长度为64位,其中高32位代表从NTP时间戳0秒至现在的秒数,低32位代表1秒以内的分数部分。

1
2
3
4
5
6
7
 0                   1                   2                   3
0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5 6 7 8 9 0 1
+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
| Seconds |
+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
| Fraction |
+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+

NTP日期格式

NTP日期格式长度为128位,其中高32位用来表示NTP时间纪元,然后用32位表示从当前纪元开始经过的秒数,最后用64位表示1秒以内的分数部分。

1
2
3
4
5
6
7
8
9
10
11
 0                   1                   2                   3
0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5 6 7 8 9 0 1
+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
| Era Number |
+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
| Era Offset |
+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
| |
| Fraction |
| |
+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+

报文格式

一个NTP v3的报文必须包含如下字段:

  • LI - Leap Indicator,2 bit整型数,指示当月最后一分钟是否包含闰秒
  • VN - Version Number,3 bit整型数,指示NTP协议的版本号。如NTP v3就是3。
  • MODE - 3 bit整型数,指示发包方的工作模式。通常来说客户端使用3(client)请求时间,服务端使用4(server)返回时间。
  • STRATUM - 8 bit整型数,代表NTP层数。0代表时钟源,如装备有GPS接收机的主服务器;1-15逐层作为下游服务器,16被定义为“无法同步”。
  • POLL - 8 bit有符号整型数,代表在间隔多少秒后再进行下一次同步。值由log2(second)计算得出。
  • PRECISION - 8 bit有符号整型数,代表系统时钟的精确度。
  • ROOT DELAY - NTP短时间格式,指示从客户端到根服务器(stratum 1的服务器)的延迟。
  • ROOT DISPERSION - NTP短时间格式,指示数据从根服务器到客户端之间可能引入的误差。
  • REFERENCE ID - 32 bit代码,用于标识一个特定的服务器,或一个参考时钟。
    • 对于stratum 0的数据包,该字段为4个ASCII字符,称作“kiss code”,用于调试和监控。
    • 对于stratum 1的数据包,该字段为参考时钟的标识符。标识符由IANA维护,此外以“X”开头的标识符都被预留给未注册的试验和开发用途。
    • 对于stratum 2~15的数据包,该字段为服务器的标识符。当服务器使用IPv4时,该字段为服务器的IP地址;当服务器使用IPv6时,该字段为IPv6地址的前四段。
  • REFERENCE TIMESTAMP - NTP时间戳格式,内容为客户端最后同步的时间。
  • ORIGIN TIMESTAMP - NTP时间格式,内容为数据包离开客户端的时间。
  • RECEIVE TIMESTAMP - NTP时间格式,内容为数据包抵达服务器的时间。
  • TRANSMIT TIMESTAMP - NTP时间格式,内容为数据包离开服务器的时间。
  • DESTINATION TIMESTAMP - NTP时间格式,内容为数据包抵达客户端的时间。
    • 注:DESTINATION TIMESTAMP并不会包含在数据包中,而是在客户端收到数据包之后,它的数值才会被确定。

那么全部组合起来,就是这个样子的:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
 0                   1                   2                   3
0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5 6 7 8 9 0 1
+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
|LI | VN |Mode | Stratum | Poll | Precision |
+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
| Root Delay |
+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
| Root Dispersion |
+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
| Reference ID |
+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
| |
+ Reference Timestamp (64) +
| |
+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
| |
+ Origin Timestamp (64) +
| |
+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
| |
+ Receive Timestamp (64) +
| |
+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
| |
+ Transmit Timestamp (64) +
| |
+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+

然而上述字段并不需要全部填写数据,实际上除了LI、VN、MODE、STRATUM之外,剩下的所有字段都可以填零。如下就是一个我用来测试的数据包:

1
2
HEX:
DB 10 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00

拆开来看的话:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
BIN:
LI = 0b11 = 3 unknown (clock unsyncronized)
VN = 0b011 = 3
MODE = 0b011 = 3 client
STRATUM = 0b00010000 = 16
POLL = 0b00000000 = 0
PRECISION = 0b00000000 = 0
ROOT DELAY = 0b00000000000000000000000000000000
ROOT DISPERSION = 0b00000000000000000000000000000000
REFERENCE ID = 00000000000000000000000000000000
REFERENCE TIMESTAMP = 00000000000000000000000000000000 00000000000000000000000000000000
ORIGIN TIMESTAMP = 00000000000000000000000000000000 00000000000000000000000000000000
RECEIVE TIMESTAMP = 00000000000000000000000000000000 00000000000000000000000000000000
TRANSMIT TIMESTAMP = 00000000000000000000000000000000 00000000000000000000000000000000

计算second和fraction

计算second很简单,取出timestamp的高32位就可以了;但是从fraction计算毫秒数比较麻烦,需要通过fraction * 10^6 / 2^32计算得到毫秒数。

这里我给出一个Java的代码片段:

1
2
3
4
5
final long seconds = (ntpTimestamp >>> 32) & 0xFFFFFFFFL;
final long secondsInMilliseconds = seconds * 1000;

final long fractionInTimestamp = (ntpTimestamp & 0xFFFFFFFFL);
final long milliseconds = fractionInTimestamp * Math.pow(10, 6) / Math.pow(2, 32);

然后计算1900年1月1日 00:00:00的UNIX时间戳作为基准UNIX时间戳,再加上secondsInMillisecondsmilliseconds,就可以得到NTP返回的当前时间了。

参考文档

  • Network Time Protocol Version 4: Protocol and Algorithms Specification - RFC
  • Network Time Protocol (NTP) 网络时间协定 - Jan Ho 的网络世界
  • The Root of All Timing: Understanding root delay and root dispersion in NTP
  • NTP Timestamp - Thompson’s Technological Insight
  • A Very Short Introduction to NTP Timestamps
  • NtpPacketUtils#getNtpTimestampMilliseconds - blackberry_time_sync_ntp - GitHub