tscns icon indicating copy to clipboard operation
tscns copied to clipboard

请教误差范围

Open xdaniel9 opened this issue 3 years ago • 7 comments

在我的服务器上,无论是程序自己calibrate还是cheat,误差都会在一分钟之内增大到微秒级甚至毫秒级 cpu是双路志强8255C,是支持constant_tsc的

另外在家里的电脑8700K上测试结果也是一样,系统都是Ubuntu 20.04.1 LTS

这里的误差是 rdtsc -> clock_gettime -> rdtsc,然后两次rdtsc结果平均值和clock_gettime结果的差

xdaniel9 avatar Jun 15 '21 04:06 xdaniel9

多路上面可能不准,通常情况下要定期校验, 多线程情况下周期性调用init 是否有问题?

include <bits/stdc++.h> 在macOS clang++上无法编译,似乎已经不是c++标准头文件了(gcc 保留)

ktprime avatar Jul 04 '21 00:07 ktprime

可以周期性重复调用calibrate(),这是多线程安全的

MengRao avatar Jul 04 '21 01:07 MengRao

我修改了下test.cc,去掉了<bits/stdc++.h>,并去掉了sleep1秒的注释,这样可以大大提高精度降低误差

MengRao avatar Jul 04 '21 03:07 MengRao

服务器跨平台使用修改rdsysns,使用steady_clock 是否比high_resolution_clock 更稳定 还是没太大区别?

改成steady_clock 后,测试数据不一样,b2c 数据误差持续变大 a: 14642789863999881, b: 14642789864006251, c: 14642789864006229, a2b: 6370, b2c: -22, good: 0, rdsysns_latency: 6339 a: 14642790864015103, b: 14642790864021136, c: 14642790864021107, a2b: 6033, b2c: -29, good: 0, rdsysns_latency: 5995 a: 14642791864033885, b: 14642791864034313, c: 14642791864034287, a2b: 428, b2c: -26, good: 0, rdsysns_latency: 393 a: 14642792864040781, b: 14642792864041466, c: 14642792864041427, a2b: 685, b2c: -39, good: 0, rdsysns_latency: 637 ... a: 14643039691289192, b: 14643039691290108, c: 14643039691289648, a2b: 916, b2c: -460, good: 0, rdsysns_latency: 447 a: 14643040691297430, b: 14643040691298408, c: 14643040691297938, a2b: 978, b2c: -470, good: 0, rdsysns_latency: 499

定期同步校验init()每次至少10ms等待, 对于服务器有点卡

ktprime avatar Jul 05 '21 02:07 ktprime

下面这段代码每200ms运行一次,为减少同步卡顿,做了如下校验不知是否有问题

void Clock::AdjustTime() const
{
    const auto nowns = tn.rdns();
    const auto nowss = tn.rdsysns(); //steady clock
    if (std::abs(nowns - nowss) > 1000)
       tn.init();
}
  1. 调整后的时钟是否因为较准往前回拨?
  2. ns_offset 变量(不同线程读写不会有同步问题?)

ktprime avatar Jul 05 '21 02:07 ktprime

只需要init一次,可以重复calibrate,这样不用等10ms,也不会有多线程问题

MengRao avatar Jul 05 '21 05:07 MengRao

笔记本上使用发现时间漂移比较严重,盒盖再翻盖后时钟误差巨大,甚至出现回拨现象,按如下解决

void Clock::AdjustTime() const
{
    const auto nowns = tn.rdns();
    const auto nowss = tn.rdsysns(); //steady clock
    if (std::abs(nowns - nowss) > 10'000) //误差比较大,频率可能发生较大变化
       tn.init();
    else if (std::abs(nowns - nowss) > 1000) //小范围内对齐
       tn.calibrate();

}

ktprime avatar Jul 14 '21 11:07 ktprime