123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330 |
- /* valid adjtimex test
- * by: John Stultz <[email protected]>
- * (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>
- #include "../kselftest.h"
- #define NSEC_PER_SEC 1000000000LL
- #define USEC_PER_SEC 1000000LL
- #define ADJ_SETOFFSET 0x0100
- #include <sys/syscall.h>
- int clock_adjtime(clockid_t id, struct timex *tx)
- {
- return syscall(__NR_clock_adjtime, id, tx);
- }
- /* 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... ");
- fflush(stdout);
- 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 set_offset(long long offset, int use_nano)
- {
- struct timex tmx = {};
- int ret;
- tmx.modes = ADJ_SETOFFSET;
- if (use_nano) {
- tmx.modes |= ADJ_NANO;
- tmx.time.tv_sec = offset / NSEC_PER_SEC;
- tmx.time.tv_usec = offset % NSEC_PER_SEC;
- if (offset < 0 && tmx.time.tv_usec) {
- tmx.time.tv_sec -= 1;
- tmx.time.tv_usec += NSEC_PER_SEC;
- }
- } else {
- tmx.time.tv_sec = offset / USEC_PER_SEC;
- tmx.time.tv_usec = offset % USEC_PER_SEC;
- if (offset < 0 && tmx.time.tv_usec) {
- tmx.time.tv_sec -= 1;
- tmx.time.tv_usec += USEC_PER_SEC;
- }
- }
- ret = clock_adjtime(CLOCK_REALTIME, &tmx);
- if (ret < 0) {
- printf("(sec: %ld usec: %ld) ", tmx.time.tv_sec, tmx.time.tv_usec);
- printf("[FAIL]\n");
- return -1;
- }
- return 0;
- }
- int set_bad_offset(long sec, long usec, int use_nano)
- {
- struct timex tmx = {};
- int ret;
- tmx.modes = ADJ_SETOFFSET;
- if (use_nano)
- tmx.modes |= ADJ_NANO;
- tmx.time.tv_sec = sec;
- tmx.time.tv_usec = usec;
- ret = clock_adjtime(CLOCK_REALTIME, &tmx);
- if (ret >= 0) {
- printf("Invalid (sec: %ld usec: %ld) did not fail! ", tmx.time.tv_sec, tmx.time.tv_usec);
- printf("[FAIL]\n");
- return -1;
- }
- return 0;
- }
- int validate_set_offset(void)
- {
- printf("Testing ADJ_SETOFFSET... ");
- fflush(stdout);
- /* Test valid values */
- if (set_offset(NSEC_PER_SEC - 1, 1))
- return -1;
- if (set_offset(-NSEC_PER_SEC + 1, 1))
- return -1;
- if (set_offset(-NSEC_PER_SEC - 1, 1))
- return -1;
- if (set_offset(5 * NSEC_PER_SEC, 1))
- return -1;
- if (set_offset(-5 * NSEC_PER_SEC, 1))
- return -1;
- if (set_offset(5 * NSEC_PER_SEC + NSEC_PER_SEC / 2, 1))
- return -1;
- if (set_offset(-5 * NSEC_PER_SEC - NSEC_PER_SEC / 2, 1))
- return -1;
- if (set_offset(USEC_PER_SEC - 1, 0))
- return -1;
- if (set_offset(-USEC_PER_SEC + 1, 0))
- return -1;
- if (set_offset(-USEC_PER_SEC - 1, 0))
- return -1;
- if (set_offset(5 * USEC_PER_SEC, 0))
- return -1;
- if (set_offset(-5 * USEC_PER_SEC, 0))
- return -1;
- if (set_offset(5 * USEC_PER_SEC + USEC_PER_SEC / 2, 0))
- return -1;
- if (set_offset(-5 * USEC_PER_SEC - USEC_PER_SEC / 2, 0))
- return -1;
- /* Test invalid values */
- if (set_bad_offset(0, -1, 1))
- return -1;
- if (set_bad_offset(0, -1, 0))
- return -1;
- if (set_bad_offset(0, 2 * NSEC_PER_SEC, 1))
- return -1;
- if (set_bad_offset(0, 2 * USEC_PER_SEC, 0))
- return -1;
- if (set_bad_offset(0, NSEC_PER_SEC, 1))
- return -1;
- if (set_bad_offset(0, USEC_PER_SEC, 0))
- return -1;
- if (set_bad_offset(0, -NSEC_PER_SEC, 1))
- return -1;
- if (set_bad_offset(0, -USEC_PER_SEC, 0))
- return -1;
- printf("[OK]\n");
- return 0;
- }
- int main(int argc, char **argv)
- {
- if (validate_freq())
- return ksft_exit_fail();
- if (validate_set_offset())
- return ksft_exit_fail();
- return ksft_exit_pass();
- }
|