Ntp服务器没有响应我的客户端

时间:2018-01-13 21:47:08

标签: c sockets ntp

我写了一个非常小的ntp客户端,但在发送数据包之后我没有收到服务器的响应,我已经用tcpdump确认了。

我已经定义了一个结构,用于传输ntp数据包。

struct NtpMessage_t {
    unsigned int first_line;
    unsigned int root_delay;
    unsigned int root_dispertion;
    unsigned int ref_id;
    unsigned long long ref_timestamp;
    unsigned long long origin_timestamp;
    unsigned long long receive_timestamp;
    unsigned long long transmit_timestamp;
};

message->first_line = 0b00100011000000000000000000000000;填充数据包后,我将数据包发送到de.ntp.pool.org,但我没有得到回复。套接字初始化以及消息的发送和接收使用以下代码完成:

for (i = result; i != NULL; i = i->ai_next) {
    cfd = socket(i->ai_family, i->ai_socktype, i->ai_protocol);

    if (cfd != -1) 
        break;
}

if (i == NULL) {
    fprintf(stderr, "Failed to create socket\n");
    exit(EXIT_FAILURE);
}

NtpMessage *message = calloc(1, sizeof(NtpMessage));
message->first_line = 0b00100011000000000000000000000000;
if (sendto(cfd, message, sizeof(NtpMessage), 0, (struct sockaddr *) i->ai_addr, i->ai_addrlen) < 0) {
    printf("Send failure: %s\n", strerror(errno));
    exit(EXIT_FAILURE);
}

if (recvfrom(cfd, message, sizeof(NtpMessage), 0, (struct sockaddr *) &from_addr, &from_addr_len) < 0) {
    printf("Receive failure: %s\n", strerror(errno));
    exit(EXIT_FAILURE);
}

Tcpdump给了我以下输出,从中我了解到我的消息正确发送但没有收到回复:

MY_IP_ADDR.50318 > 64.99.80.121.123: [udp sum ok] NTPv0, length 48
unspecified, Leap indicator:  (0), Stratum 0 (unspecified), poll 0 (1s), precision 35
Root Delay: 0.000000, Root dispersion: 0.000000, Reference-ID: (unspec)
  Reference Timestamp:  0.000000000
  Originator Timestamp: 0.000000000
  Receive Timestamp:    0.000000000
  Transmit Timestamp:   0.000000000
    Originator - Receive Timestamp:  0.000000000
    Originator - Transmit Timestamp: 0.000000000

由于在另一个方向没有消息,这意味着没有回应,如果我没有弄错的话。是这样,如果是的话,这里出了什么问题?我的代码有问题吗?

1 个答案:

答案 0 :(得分:0)

服务器不能发出对您的ntp数据包的响应可能有很多原因。我看到你在示例中隐藏了几件事,所以很可能问题是对NTP协议规范的无知。

如何填写数据包内容?您是否检查过编译器备用字节以填充结构对齐的漏洞?您是否考虑过,在NTP协议中,所有short,int和64bit时间戳都必须按网络字节顺序排列(最重要的字节优先)????

最后,NTP服务器通过忽略传入的数据包来处理不正确的输入,但您认为数据包是正确的。