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 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 219 220 221 222 223 224 225 226 227 228 229 230 231 232 233 234 235 236 237 238 239 240 241 242 243 244 245 246 247 248 249 250 251 252 253 254 255 256 257 258 259 260 261 262 263 264 265 266 267 268 269 270 271 272 273 274 275 276 277 278 279 280 281 282 283 284 285 286 287 288 289 290 291 292 293 294 295 296 297 298 299 300 301 302 303 304 305 306 307 308 309 310 311 312
| #include <sys/timex.h> #include <stdio.h> #include <stdlib.h> #include <string.h> #include <errno.h> #include <time.h> #include <unistd.h> #include <math.h>
/** * NTP服务器信息结构 */ typedef struct { char hostname[256]; int port; double stratum; // 层级 double delay; // 延迟 double offset; // 偏移 double dispersion; // 离散度 time_t last_contact; // 最后联系时间 int reachable; // 是否可达 } ntp_server_t;
/** * NTP客户端结构 */ typedef struct { ntp_server_t servers[5]; int server_count; int current_server; double sync_threshold; // 同步阈值(毫秒) double max_adjustment; // 最大调整值(毫秒) int sync_interval; // 同步间隔(秒) } ntp_client_t;
/** * 初始化NTP客户端 */ int init_ntp_client(ntp_client_t *client) { memset(client, 0, sizeof(ntp_client_t)); // 初始化测试服务器 const char *test_servers[] = { "pool.ntp.org", "time.google.com", "time.cloudflare.com", "ntp.aliyun.com", NULL }; printf("=== NTP客户端初始化 ===\n"); for (int i = 0; test_servers[i] && i < 5; i++) { ntp_server_t *server = &client->servers[i]; strncpy(server->hostname, test_servers[i], sizeof(server->hostname) - 1); server->hostname[sizeof(server->hostname) - 1] = '\0'; server->port = 123; // NTP标准端口 server->stratum = 2 + i; // 模拟不同层级 server->delay = 0.05 + (rand() / (double)RAND_MAX) * 0.1; // 50-150ms延迟 server->offset = (rand() / (double)RAND_MAX) * 2.0 - 1.0; // -1到1秒偏移 server->dispersion = 0.01 + (rand() / (double)RAND_MAX) * 0.05; // 10-60ms离散度 server->last_contact = time(NULL); server->reachable = 1; // 模拟可达 client->server_count++; printf(" 添加服务器 %d: %s (层级: %.0f)\n", i + 1, server->hostname, server->stratum); } client->current_server = 0; client->sync_threshold = 100.0; // 100ms阈值 client->max_adjustment = 500.0; // 500ms最大调整 client->sync_interval = 60; // 60秒同步间隔 printf(" 同步阈值: %.1f ms\n", client->sync_threshold); printf(" 最大调整: %.1f ms\n", client->max_adjustment); printf(" 同步间隔: %d 秒\n", client->sync_interval); return 0; }
/** * 选择最佳NTP服务器 */ int select_best_ntp_server(ntp_client_t *client) { int best_server = -1; double best_quality = 999999.0; printf("选择最佳NTP服务器:\n"); for (int i = 0; i < client->server_count; i++) { ntp_server_t *server = &client->servers[i]; if (server->reachable) { // 计算服务器质量(基于层级、延迟和离散度) double quality = server->stratum + server->delay * 10 + server->dispersion * 5; printf(" 服务器 %d (%s): 质量评分 %.3f\n", i + 1, server->hostname, quality); if (quality < best_quality) { best_quality = quality; best_server = i; } } } if (best_server != -1) { client->current_server = best_server; printf(" 选择最佳服务器: %s\n", client->servers[best_server].hostname); } return best_server; }
/** * 模拟NTP时间同步 */ int simulate_ntp_sync(ntp_client_t *client) { struct timex tx; int result; int best_server = select_best_ntp_server(client); if (best_server == -1) { printf("没有可用的NTP服务器\n"); return -1; } ntp_server_t *server = &client->servers[best_server]; printf("=== NTP时间同步 ===\n"); printf("同步服务器: %s\n", server->hostname); printf("服务器层级: %.0f\n", server->stratum); printf("网络延迟: %.3f 秒\n", server->delay); printf("时钟偏移: %.3f 秒\n", server->offset); printf("离散度: %.3f 秒\n", server->dispersion); // 检查是否需要同步 if (fabs(server->offset) * 1000 > client->sync_threshold) { printf("时钟偏移过大,需要同步\n"); // 如果有root权限,进行时钟调整 if (getuid() == 0) { printf("具有root权限,进行时钟调整:\n"); memset(&tx, 0, sizeof(tx)); // 根据偏移大小选择调整策略 if (fabs(server->offset) > 1.0) { // 大偏移:步进调整 printf(" 大偏移调整:\n"); tx.modes = ADJ_SETOFFSET; tx.time.tv_sec = (long)server->offset; tx.time.tv_usec = (long)((server->offset - (long)server->offset) * 1000000); printf(" 设置时间偏移: %ld.%06ld 秒\n", tx.time.tv_sec, tx.time.tv_usec); } else { // 小偏移:渐进调整 printf(" 渐进调整:\n"); tx.modes = ADJ_OFFSET | ADJ_STATUS; tx.offset = (long)(server->offset * 1000000); // 转换为微秒 tx.status = STA_PLL; printf(" 调整偏移: %ld 微秒\n", tx.offset); } result = adjtimex(&tx); if (result == -1) { printf(" 时钟调整失败: %s\n", strerror(errno)); return -1; } printf(" ✓ 时钟调整成功\n"); // 显示调整后的状态 printf(" 调整后状态: "); switch (result) { case TIME_OK: printf("TIME_OK\n"); break; case TIME_INS: printf("TIME_INS\n"); break; case TIME_DEL: printf("TIME_DEL\n"); break; case TIME_OOP: printf("TIME_OOP\n"); break; case TIME_WAIT: printf("TIME_WAIT\n"); break; case TIME_ERROR: printf("TIME_ERROR\n"); break; default: printf("状态 %d\n", result); break; } } else { printf("没有root权限,无法进行时钟调整\n"); printf("建议使用NTP守护进程进行时间同步\n"); } } else { printf("时钟偏移在可接受范围内,无需调整\n"); } // 更新服务器联系时间 server->last_contact = time(NULL); return 0; }
/** * 演示NTP客户端功能 */ int demo_ntp_client() { ntp_client_t client; struct timex tx; int result; printf("=== NTP客户端功能演示 ===\n"); // 初始化客户端 printf("1. 初始化NTP客户端:\n"); if (init_ntp_client(&client) != 0) { return -1; } // 显示当前系统时间 printf("\n2. 当前系统时间:\n"); struct timeval current_time; gettimeofday(¤t_time, NULL); printf(" 系统时间: %ld.%06ld\n", current_time.tv_sec, current_time.tv_usec); // 获取当前时钟状态 printf("\n3. 当前时钟状态:\n"); memset(&tx, 0, sizeof(tx)); tx.modes = 0; // 查询模式 result = adjtimex(&tx); if (result != -1) { printf(" 时钟状态: "); switch (tx.state) { case TIME_OK: printf("TIME_OK (正常)\n"); break; case TIME_ERROR: printf("TIME_ERROR (错误)\n"); break; default: printf("状态 %d\n", tx.state); break; } printf(" 时钟偏移: %ld.%06ld 秒\n", tx.offset / 1000000, labs(tx.offset) % 1000000); printf(" 时钟频率: %.3f ppm\n", (double)tx.freq / 65536.0); printf(" 最大误差: %ld 毫秒\n", tx.maxerror); printf(" 估算误差: %ld 毫秒\n", tx.esterror); } // 模拟NTP同步 printf("\n4. 模拟NTP时间同步:\n"); if (simulate_ntp_sync(&client) != 0) { printf("NTP同步失败\n"); return -1; } // 显示同步后状态 printf("\n5. 同步后时钟状态:\n"); memset(&tx, 0, sizeof(tx)); tx.modes = 0; // 查询模式 result = adjtimex(&tx); if (result != -1) { printf(" 时钟状态: "); switch (result) { case TIME_OK: printf("TIME_OK (正常)\n"); break; case TIME_INS: printf("TIME_INS (即将插入闰秒)\n"); break; case TIME_DEL: printf("TIME_DEL (即将删除闰秒)\n"); break; case TIME_OOP: printf("TIME_OOP (闰秒处理中)\n"); break; case TIME_WAIT: printf("TIME_WAIT (等待同步)\n"); break; case TIME_ERROR: printf("TIME_ERROR (时钟错误)\n"); break; default: printf("状态 %d\n", result); break; } printf(" 时钟偏移: %ld.%06ld 秒\n", tx.offset / 1000000, labs(tx.offset) % 1000000); printf(" 时钟频率: %.3f ppm\n", (double)tx.freq / 65536.0); printf(" 最大误差: %ld 毫秒\n", tx.maxerror); printf(" 估算误差: %ld 毫秒\n", tx.esterror); } // 显示NTP服务器信息 printf("\n6. NTP服务器信息:\n"); for (int i = 0; i < client.server_count; i++) { ntp_server_t *server = &client.servers[i]; printf(" 服务器 %d: %s\n", i + 1, server->hostname); printf(" 层级: %.0f\n", server->stratum); printf(" 延迟: %.3f 秒\n", server->delay); printf(" 偏移: %.3f 秒\n", server->offset); printf(" 离散度: %.3f 秒\n", server->dispersion); printf(" 最后联系: %s", ctime(&server->last_contact)); printf(" 可达: %s\n", server->reachable ? "是" : "否"); } // 显示NTP客户端优势 printf("\n=== NTP客户端优势 ===\n"); printf("1. 高精度同步:\n"); printf(" ✓ 微秒级时间精度\n"); printf(" ✓ PPM级频率控制\n"); printf(" ✓ 毫秒级偏移调整\n"); printf("\n2. 智能选择:\n"); printf(" ✓ 多服务器支持\n"); printf(" ✓ 质量评分算法\n"); printf(" ✓ 动态服务器切换\n"); printf("\n3. 安全特性:\n"); printf(" ✓ 权限检查\n"); printf(" ✓ 错误处理\n"); printf(" ✓ 状态监控\n"); printf("\n4. 灵活配置:\n"); printf(" ✓ 可配置阈值\n"); printf(" ✓ 动态间隔调整\n"); printf(" ✓ 多种调整策略\n"); return 0; }
int main() { return demo_ntp_client(); }
|