| #include "libcflat.h" |
| #include "smp.h" |
| #include "atomic.h" |
| #include "processor.h" |
| #include "kvmclock.h" |
| |
| #define DEFAULT_TEST_LOOPS 100000000L |
| #define DEFAULT_THRESHOLD 5L |
| |
| long loops = DEFAULT_TEST_LOOPS; |
| long sec = 0; |
| long threshold = DEFAULT_THRESHOLD; |
| |
| struct test_info { |
| struct spinlock lock; |
| u64 warps; /* warp count */ |
| u64 stalls; /* stall count */ |
| long long worst; /* worst warp */ |
| volatile cycle_t last; /* last cycle seen by test */ |
| int check; /* check cycle ? */ |
| }; |
| |
| struct test_info ti[4]; |
| |
| static void wallclock_test(void *data) |
| { |
| int *p_err = data; |
| long ksec, offset; |
| struct timespec ts; |
| |
| kvm_get_wallclock(&ts); |
| ksec = ts.tv_sec; |
| |
| offset = ksec - sec; |
| printf("Raw nanoseconds value from kvmclock: %" PRIu64 " (cpu %d)\n", kvm_clock_read(), smp_id()); |
| printf("Seconds get from kvmclock: %ld (cpu %d, offset: %ld)\n", ksec, smp_id(), offset); |
| |
| if (offset > threshold || offset < -threshold) { |
| printf("offset too large!\n"); |
| (*p_err)++; |
| } |
| } |
| |
| static void kvm_clock_test(void *data) |
| { |
| struct test_info *hv_test_info = (struct test_info *)data; |
| long i, check = hv_test_info->check; |
| |
| for (i = 0; i < loops; i++){ |
| cycle_t t0, t1; |
| long long delta; |
| |
| if (check == 0) { |
| kvm_clock_read(); |
| continue; |
| } |
| |
| spin_lock(&hv_test_info->lock); |
| t1 = kvm_clock_read(); |
| t0 = hv_test_info->last; |
| hv_test_info->last = kvm_clock_read(); |
| spin_unlock(&hv_test_info->lock); |
| |
| delta = t1 - t0; |
| if (delta < 0) { |
| spin_lock(&hv_test_info->lock); |
| ++hv_test_info->warps; |
| if (delta < hv_test_info->worst){ |
| hv_test_info->worst = delta; |
| printf("Worst warp %lld\n", hv_test_info->worst); |
| } |
| spin_unlock(&hv_test_info->lock); |
| } |
| if (delta == 0) |
| ++hv_test_info->stalls; |
| |
| if (!((unsigned long)i & 31)) |
| asm volatile("rep; nop"); |
| } |
| } |
| |
| static int cycle_test(int check, struct test_info *ti) |
| { |
| unsigned long long begin, end; |
| |
| begin = rdtsc(); |
| |
| ti->check = check; |
| on_cpus(kvm_clock_test, ti); |
| |
| end = rdtsc(); |
| |
| printf("Total vcpus: %d\n", cpu_count()); |
| printf("Test loops: %ld\n", loops); |
| if (check == 1) { |
| printf("Total warps: %" PRId64 "\n", ti->warps); |
| printf("Total stalls: %" PRId64 "\n", ti->stalls); |
| printf("Worst warp: %lld\n", ti->worst); |
| } else |
| printf("TSC cycles: %lld\n", end - begin); |
| |
| return ti->warps ? 1 : 0; |
| } |
| |
| int main(int ac, char **av) |
| { |
| int nerr = 0; |
| int ncpus; |
| int i; |
| |
| if (ac > 1) |
| loops = atol(av[1]); |
| if (ac > 2) |
| sec = atol(av[2]); |
| if (ac > 3) |
| threshold = atol(av[3]); |
| |
| ncpus = cpu_count(); |
| if (ncpus > MAX_CPU) |
| report_abort("number cpus exceeds %d", MAX_CPU); |
| |
| on_cpus(kvm_clock_init, NULL); |
| |
| if (ac > 2) { |
| printf("Wallclock test, threshold %ld\n", threshold); |
| printf("Seconds get from host: %ld\n", sec); |
| for (i = 0; i < ncpus; ++i) |
| on_cpu(i, wallclock_test, &nerr); |
| } |
| |
| printf("Check the stability of raw cycle ...\n"); |
| pvclock_set_flags(PVCLOCK_TSC_STABLE_BIT |
| | PVCLOCK_RAW_CYCLE_BIT); |
| if (cycle_test(1, &ti[0])) |
| printf("Raw cycle is not stable\n"); |
| else |
| printf("Raw cycle is stable\n"); |
| |
| pvclock_set_flags(PVCLOCK_TSC_STABLE_BIT); |
| printf("Monotonic cycle test:\n"); |
| nerr += cycle_test(1, &ti[1]); |
| |
| printf("Measure the performance of raw cycle ...\n"); |
| pvclock_set_flags(PVCLOCK_TSC_STABLE_BIT |
| | PVCLOCK_RAW_CYCLE_BIT); |
| cycle_test(0, &ti[2]); |
| |
| printf("Measure the performance of adjusted cycle ...\n"); |
| pvclock_set_flags(PVCLOCK_TSC_STABLE_BIT); |
| cycle_test(0, &ti[3]); |
| |
| on_cpus(kvm_clock_clear, NULL); |
| |
| return nerr > 0 ? 1 : 0; |
| } |