/* See LICENSE file for copyright and license details. */ #include "libsimple.h" #ifndef TEST int libsimple_difftimeval(struct timeval *diff, const struct timeval *minuend, const struct timeval *subtrahend) { time_t s; if (LIBSIMPLE_SSUB_OVERFLOW(minuend->tv_sec, subtrahend->tv_sec, &s, TIME_MIN, TIME_MAX)) { if (subtrahend->tv_sec < 0) goto too_large; else goto too_small; } diff->tv_usec = minuend->tv_usec - subtrahend->tv_usec; if (diff->tv_usec < 0) { if (LIBSIMPLE_SDECR_OVERFLOW(&s, TIME_MIN)) goto too_small; diff->tv_usec += 1000000L; } diff->tv_sec = s; return 0; too_large: diff->tv_sec = TIME_MAX; diff->tv_usec = 999999L; errno = ERANGE; return -1; too_small: diff->tv_sec = TIME_MIN; diff->tv_usec = 0; errno = ERANGE; return -1; } #else #include "test.h" int main(void) { struct timeval r, a, b; a.tv_sec = 0, a.tv_usec = 0L; b.tv_sec = 0, b.tv_usec = 0L; assert(!libsimple_difftimeval(&r, &a, &b)); assert(r.tv_sec == 0); assert(r.tv_usec == 0L); a.tv_sec = 10, a.tv_usec = 0L; b.tv_sec = 0, b.tv_usec = 0L; assert(!libsimple_difftimeval(&r, &a, &b)); assert(r.tv_sec == 10); assert(r.tv_usec == 0L); a.tv_sec = 10, a.tv_usec = 100L; b.tv_sec = 0, b.tv_usec = 0L; assert(!libsimple_difftimeval(&r, &a, &b)); assert(r.tv_sec == 10); assert(r.tv_usec == 100L); a.tv_sec = 10, a.tv_usec = 100L; b.tv_sec = 1, b.tv_usec = 0L; assert(!libsimple_difftimeval(&r, &a, &b)); assert(r.tv_sec == 9); assert(r.tv_usec == 100L); a.tv_sec = 10, a.tv_usec = 100L; b.tv_sec = 0, b.tv_usec = 1L; assert(!libsimple_difftimeval(&r, &a, &b)); assert(r.tv_sec == 10); assert(r.tv_usec == 99L); a.tv_sec = -10, a.tv_usec = 100L; b.tv_sec = 0, b.tv_usec = 0L; assert(!libsimple_difftimeval(&r, &a, &b)); assert(r.tv_sec == -10); assert(r.tv_usec == 100L); a.tv_sec = 10, a.tv_usec = 1L; b.tv_sec = 1, b.tv_usec = 100L; assert(!libsimple_difftimeval(&r, &a, &b)); assert(r.tv_sec == 8); assert(r.tv_usec == 1000001L - 100L); a.tv_sec = 10, a.tv_usec = 0L; b.tv_sec = 20, b.tv_usec = 0L; assert(!libsimple_difftimeval(&r, &a, &b)); assert(r.tv_sec == -10); assert(r.tv_usec == 0L); a.tv_sec = 10, a.tv_usec = 10L; b.tv_sec = 20, b.tv_usec = 0L; assert(!libsimple_difftimeval(&r, &a, &b)); assert(r.tv_sec == -10); assert(r.tv_usec == 10L); a.tv_sec = 10, a.tv_usec = 10L; b.tv_sec = 20, b.tv_usec = 20L; assert(!libsimple_difftimeval(&r, &a, &b)); assert(r.tv_sec == -10 - 1); assert(r.tv_usec == 1000010L - 20L); a.tv_sec = 10, a.tv_usec = 10L; b.tv_sec = -20, b.tv_usec = 0L; assert(!libsimple_difftimeval(&r, &a, &b)); assert(r.tv_sec == 30); assert(r.tv_usec == 10L); a.tv_sec = 0, a.tv_usec = 10L; b.tv_sec = 0, b.tv_usec = 0L; assert(!libsimple_difftimeval(&r, &a, &b)); assert(r.tv_sec == 0); assert(r.tv_usec == 10L); a.tv_sec = 0, a.tv_usec = 0L; b.tv_sec = 0, b.tv_usec = 10L; assert(!libsimple_difftimeval(&r, &a, &b)); assert(r.tv_sec == -1); assert(r.tv_usec == 1000000L - 10L); a.tv_sec = TIME_MIN, a.tv_usec = 0L; b.tv_sec = 0, b.tv_usec = 0L; assert(!libsimple_difftimeval(&r, &a, &b)); assert(r.tv_sec == TIME_MIN); assert(r.tv_usec == 0L); a.tv_sec = TIME_MIN, a.tv_usec = 0L; b.tv_sec = 0, b.tv_usec = 1L; assert(libsimple_difftimeval(&r, &a, &b) == -1 && errno == ERANGE); assert(r.tv_sec == TIME_MIN); assert(r.tv_usec == 0L); a.tv_sec = TIME_MIN, a.tv_usec = 0L; b.tv_sec = -1, b.tv_usec = 1L; assert(!libsimple_difftimeval(&r, &a, &b)); assert(r.tv_sec == TIME_MIN); assert(r.tv_usec == 999999L); a.tv_sec = TIME_MAX, a.tv_usec = 0L; b.tv_sec = -1, b.tv_usec = 0L; assert(libsimple_difftimeval(&r, &a, &b) == -1 && errno == ERANGE); assert(r.tv_sec == TIME_MAX); assert(r.tv_usec == 999999L); return 0; } #endif