| /* valid adjtimex test |
| * by: John Stultz <john.stultz@linaro.org> |
| * (C) Copyright Linaro 2015 |
| * Licensed under the GPLv2 |
| * |
| * This test validates adjtimex interface with valid |
| * and invalid test data. |
| * |
| * Usage: valid-adjtimex |
| * |
| * To build: |
| * $ gcc valid-adjtimex.c -o valid-adjtimex -lrt |
| * |
| * This program is free software: you can redistribute it and/or modify |
| * it under the terms of the GNU General Public License as published by |
| * the Free Software Foundation, either version 2 of the License, or |
| * (at your option) any later version. |
| * |
| * This program is distributed in the hope that it will be useful, |
| * but WITHOUT ANY WARRANTY; without even the implied warranty of |
| * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the |
| * GNU General Public License for more details. |
| */ |
| |
| |
| |
| #include <stdio.h> |
| #include <stdlib.h> |
| #include <time.h> |
| #include <sys/time.h> |
| #include <sys/timex.h> |
| #include <string.h> |
| #include <signal.h> |
| #include <unistd.h> |
| #ifdef KTEST |
| #include "../kselftest.h" |
| #else |
| static inline int ksft_exit_pass(void) |
| { |
| exit(0); |
| } |
| static inline int ksft_exit_fail(void) |
| { |
| exit(1); |
| } |
| #endif |
| |
| #define NSEC_PER_SEC 1000000000L |
| |
| /* clear NTP time_status & time_state */ |
| int clear_time_state(void) |
| { |
| struct timex tx; |
| int ret; |
| |
| tx.modes = ADJ_STATUS; |
| tx.status = 0; |
| ret = adjtimex(&tx); |
| return ret; |
| } |
| |
| #define NUM_FREQ_VALID 32 |
| #define NUM_FREQ_OUTOFRANGE 4 |
| #define NUM_FREQ_INVALID 2 |
| |
| long valid_freq[NUM_FREQ_VALID] = { |
| -499<<16, |
| -450<<16, |
| -400<<16, |
| -350<<16, |
| -300<<16, |
| -250<<16, |
| -200<<16, |
| -150<<16, |
| -100<<16, |
| -75<<16, |
| -50<<16, |
| -25<<16, |
| -10<<16, |
| -5<<16, |
| -1<<16, |
| -1000, |
| 1<<16, |
| 5<<16, |
| 10<<16, |
| 25<<16, |
| 50<<16, |
| 75<<16, |
| 100<<16, |
| 150<<16, |
| 200<<16, |
| 250<<16, |
| 300<<16, |
| 350<<16, |
| 400<<16, |
| 450<<16, |
| 499<<16, |
| }; |
| |
| long outofrange_freq[NUM_FREQ_OUTOFRANGE] = { |
| -1000<<16, |
| -550<<16, |
| 550<<16, |
| 1000<<16, |
| }; |
| |
| #define LONG_MAX (~0UL>>1) |
| #define LONG_MIN (-LONG_MAX - 1) |
| |
| long invalid_freq[NUM_FREQ_INVALID] = { |
| LONG_MAX, |
| LONG_MIN, |
| }; |
| |
| int validate_freq(void) |
| { |
| struct timex tx; |
| int ret, pass = 0; |
| int i; |
| |
| clear_time_state(); |
| |
| memset(&tx, 0, sizeof(struct timex)); |
| /* Set the leap second insert flag */ |
| |
| printf("Testing ADJ_FREQ... "); |
| for (i = 0; i < NUM_FREQ_VALID; i++) { |
| tx.modes = ADJ_FREQUENCY; |
| tx.freq = valid_freq[i]; |
| |
| ret = adjtimex(&tx); |
| if (ret < 0) { |
| printf("[FAIL]\n"); |
| printf("Error: adjtimex(ADJ_FREQ, %ld - %ld ppm\n", |
| valid_freq[i], valid_freq[i]>>16); |
| pass = -1; |
| goto out; |
| } |
| tx.modes = 0; |
| ret = adjtimex(&tx); |
| if (tx.freq != valid_freq[i]) { |
| printf("Warning: freq value %ld not what we set it (%ld)!\n", |
| tx.freq, valid_freq[i]); |
| } |
| } |
| for (i = 0; i < NUM_FREQ_OUTOFRANGE; i++) { |
| tx.modes = ADJ_FREQUENCY; |
| tx.freq = outofrange_freq[i]; |
| |
| ret = adjtimex(&tx); |
| if (ret < 0) { |
| printf("[FAIL]\n"); |
| printf("Error: adjtimex(ADJ_FREQ, %ld - %ld ppm\n", |
| outofrange_freq[i], outofrange_freq[i]>>16); |
| pass = -1; |
| goto out; |
| } |
| tx.modes = 0; |
| ret = adjtimex(&tx); |
| if (tx.freq == outofrange_freq[i]) { |
| printf("[FAIL]\n"); |
| printf("ERROR: out of range value %ld actually set!\n", |
| tx.freq); |
| pass = -1; |
| goto out; |
| } |
| } |
| |
| |
| if (sizeof(long) == 8) { /* this case only applies to 64bit systems */ |
| for (i = 0; i < NUM_FREQ_INVALID; i++) { |
| tx.modes = ADJ_FREQUENCY; |
| tx.freq = invalid_freq[i]; |
| ret = adjtimex(&tx); |
| if (ret >= 0) { |
| printf("[FAIL]\n"); |
| printf("Error: No failure on invalid ADJ_FREQUENCY %ld\n", |
| invalid_freq[i]); |
| pass = -1; |
| goto out; |
| } |
| } |
| } |
| |
| printf("[OK]\n"); |
| out: |
| /* reset freq to zero */ |
| tx.modes = ADJ_FREQUENCY; |
| tx.freq = 0; |
| ret = adjtimex(&tx); |
| |
| return pass; |
| } |
| |
| |
| int main(int argc, char **argv) |
| { |
| if (validate_freq()) |
| return ksft_exit_fail(); |
| |
| return ksft_exit_pass(); |
| } |