mirror of https://github.com/openssl/openssl.git
				
				
				
			
		
			
				
	
	
		
			244 lines
		
	
	
		
			5.8 KiB
		
	
	
	
		
			C
		
	
	
	
			
		
		
	
	
			244 lines
		
	
	
		
			5.8 KiB
		
	
	
	
		
			C
		
	
	
	
/*
 | 
						|
 * Copyright 2022-2023 The OpenSSL Project Authors. All Rights Reserved.
 | 
						|
 *
 | 
						|
 * Licensed under the Apache License 2.0 (the "License").  You may not use
 | 
						|
 * this file except in compliance with the License.  You can obtain a copy
 | 
						|
 * in the file LICENSE in the source distribution or at
 | 
						|
 * https://www.openssl.org/source/license.html
 | 
						|
 */
 | 
						|
 | 
						|
#ifndef OSSL_INTERNAL_TIME_H
 | 
						|
# define OSSL_INTERNAL_TIME_H
 | 
						|
# pragma once
 | 
						|
 | 
						|
# include <openssl/e_os2.h>     /* uint64_t */
 | 
						|
# include "internal/e_os.h"
 | 
						|
# include "internal/e_winsock.h" /* for struct timeval */
 | 
						|
# include "internal/safe_math.h"
 | 
						|
 | 
						|
/*
 | 
						|
 * Internal type defining a time.
 | 
						|
 * This should be treated as an opaque structure.
 | 
						|
 *
 | 
						|
 * The time datum is Unix's 1970 and at nanosecond precision, this gives
 | 
						|
 * a range of 584 years roughly.
 | 
						|
 */
 | 
						|
typedef struct {
 | 
						|
    uint64_t t;     /* Ticks since the epoch */
 | 
						|
} OSSL_TIME;
 | 
						|
 | 
						|
/* The precision of times allows this many values per second */
 | 
						|
# define OSSL_TIME_SECOND ((uint64_t)1000000000)
 | 
						|
 | 
						|
/* One millisecond. */
 | 
						|
# define OSSL_TIME_MS     (OSSL_TIME_SECOND / 1000)
 | 
						|
 | 
						|
/* One microsecond. */
 | 
						|
# define OSSL_TIME_US     (OSSL_TIME_MS     / 1000)
 | 
						|
 | 
						|
/* One nanosecond. */
 | 
						|
# define OSSL_TIME_NS     (OSSL_TIME_US     / 1000)
 | 
						|
 | 
						|
#define ossl_seconds2time(s) ossl_ticks2time((s) * OSSL_TIME_SECOND)
 | 
						|
#define ossl_time2seconds(t) (ossl_time2ticks(t) / OSSL_TIME_SECOND)
 | 
						|
#define ossl_ms2time(ms) ossl_ticks2time((ms) * OSSL_TIME_MS)
 | 
						|
#define ossl_time2ms(t) (ossl_time2ticks(t) / OSSL_TIME_MS)
 | 
						|
#define ossl_us2time(us) ossl_ticks2time((us) * OSSL_TIME_US)
 | 
						|
#define ossl_time2us(t) (ossl_time2ticks(t) / OSSL_TIME_US)
 | 
						|
 | 
						|
/*
 | 
						|
 * Arithmetic operations on times.
 | 
						|
 * These operations are saturating, in that an overflow or underflow returns
 | 
						|
 * the largest or smallest value respectively.
 | 
						|
 */
 | 
						|
OSSL_SAFE_MATH_UNSIGNED(time, uint64_t)
 | 
						|
 | 
						|
/* Convert a tick count into a time */
 | 
						|
static ossl_unused ossl_inline
 | 
						|
OSSL_TIME ossl_ticks2time(uint64_t ticks)
 | 
						|
{
 | 
						|
    OSSL_TIME r;
 | 
						|
 | 
						|
    r.t = ticks;
 | 
						|
    return r;
 | 
						|
}
 | 
						|
 | 
						|
/* Convert a time to a tick count */
 | 
						|
static ossl_unused ossl_inline
 | 
						|
uint64_t ossl_time2ticks(OSSL_TIME t)
 | 
						|
{
 | 
						|
    return t.t;
 | 
						|
}
 | 
						|
 | 
						|
/* Get current time */
 | 
						|
OSSL_TIME ossl_time_now(void);
 | 
						|
 | 
						|
/* The beginning and end of the time range */
 | 
						|
static ossl_unused ossl_inline
 | 
						|
OSSL_TIME ossl_time_zero(void)
 | 
						|
{
 | 
						|
    return ossl_ticks2time(0);
 | 
						|
}
 | 
						|
 | 
						|
static ossl_unused ossl_inline
 | 
						|
OSSL_TIME ossl_time_infinite(void)
 | 
						|
{
 | 
						|
    return ossl_ticks2time(~(uint64_t)0);
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
/* Convert time to timeval */
 | 
						|
static ossl_unused ossl_inline
 | 
						|
struct timeval ossl_time_to_timeval(OSSL_TIME t)
 | 
						|
{
 | 
						|
    struct timeval tv;
 | 
						|
    int err = 0;
 | 
						|
 | 
						|
    /*
 | 
						|
     * Round up any nano secs which struct timeval doesn't support. Ensures that
 | 
						|
     * we never return a zero time if the input time is non zero
 | 
						|
     */
 | 
						|
    t.t = safe_add_time(t.t, OSSL_TIME_US - 1, &err);
 | 
						|
    if (err)
 | 
						|
        t = ossl_time_infinite();
 | 
						|
 | 
						|
#ifdef _WIN32
 | 
						|
    tv.tv_sec = (long int)(t.t / OSSL_TIME_SECOND);
 | 
						|
#else
 | 
						|
    tv.tv_sec = (time_t)(t.t / OSSL_TIME_SECOND);
 | 
						|
#endif
 | 
						|
    tv.tv_usec = (t.t % OSSL_TIME_SECOND) / OSSL_TIME_US;
 | 
						|
    return tv;
 | 
						|
}
 | 
						|
 | 
						|
/* Convert timeval to time */
 | 
						|
static ossl_unused ossl_inline
 | 
						|
OSSL_TIME ossl_time_from_timeval(struct timeval tv)
 | 
						|
{
 | 
						|
    OSSL_TIME t;
 | 
						|
 | 
						|
#ifndef __DJGPP__ /* tv_sec is unsigned on djgpp. */
 | 
						|
    if (tv.tv_sec < 0)
 | 
						|
        return ossl_time_zero();
 | 
						|
#endif
 | 
						|
    t.t = tv.tv_sec * OSSL_TIME_SECOND + tv.tv_usec * OSSL_TIME_US;
 | 
						|
    return t;
 | 
						|
}
 | 
						|
 | 
						|
/* Convert OSSL_TIME to time_t */
 | 
						|
static ossl_unused ossl_inline
 | 
						|
time_t ossl_time_to_time_t(OSSL_TIME t)
 | 
						|
{
 | 
						|
    return (time_t)(t.t / OSSL_TIME_SECOND);
 | 
						|
}
 | 
						|
 | 
						|
/* Convert time_t to OSSL_TIME */
 | 
						|
static ossl_unused ossl_inline
 | 
						|
OSSL_TIME ossl_time_from_time_t(time_t t)
 | 
						|
{
 | 
						|
    OSSL_TIME ot;
 | 
						|
 | 
						|
    ot.t = t;
 | 
						|
    ot.t *= OSSL_TIME_SECOND;
 | 
						|
    return ot;
 | 
						|
}
 | 
						|
 | 
						|
/* Compare two time values, return -1 if less, 1 if greater and 0 if equal */
 | 
						|
static ossl_unused ossl_inline
 | 
						|
int ossl_time_compare(OSSL_TIME a, OSSL_TIME b)
 | 
						|
{
 | 
						|
    if (a.t > b.t)
 | 
						|
        return 1;
 | 
						|
    if (a.t < b.t)
 | 
						|
        return -1;
 | 
						|
    return 0;
 | 
						|
}
 | 
						|
 | 
						|
/* Returns true if an OSSL_TIME is ossl_time_zero(). */
 | 
						|
static ossl_unused ossl_inline
 | 
						|
int ossl_time_is_zero(OSSL_TIME t)
 | 
						|
{
 | 
						|
    return ossl_time_compare(t, ossl_time_zero()) == 0;
 | 
						|
}
 | 
						|
 | 
						|
/* Returns true if an OSSL_TIME is ossl_time_infinite(). */
 | 
						|
static ossl_unused ossl_inline
 | 
						|
int ossl_time_is_infinite(OSSL_TIME t)
 | 
						|
{
 | 
						|
    return ossl_time_compare(t, ossl_time_infinite()) == 0;
 | 
						|
}
 | 
						|
 | 
						|
static ossl_unused ossl_inline
 | 
						|
OSSL_TIME ossl_time_add(OSSL_TIME a, OSSL_TIME b)
 | 
						|
{
 | 
						|
    OSSL_TIME r;
 | 
						|
    int err = 0;
 | 
						|
 | 
						|
    r.t = safe_add_time(a.t, b.t, &err);
 | 
						|
    return err ? ossl_time_infinite() : r;
 | 
						|
}
 | 
						|
 | 
						|
static ossl_unused ossl_inline
 | 
						|
OSSL_TIME ossl_time_subtract(OSSL_TIME a, OSSL_TIME b)
 | 
						|
{
 | 
						|
    OSSL_TIME r;
 | 
						|
    int err = 0;
 | 
						|
 | 
						|
    r.t = safe_sub_time(a.t, b.t, &err);
 | 
						|
    return err ? ossl_time_zero() : r;
 | 
						|
}
 | 
						|
 | 
						|
/* Returns |a - b|. */
 | 
						|
static ossl_unused ossl_inline
 | 
						|
OSSL_TIME ossl_time_abs_difference(OSSL_TIME a, OSSL_TIME b)
 | 
						|
{
 | 
						|
    return a.t > b.t ? ossl_time_subtract(a, b)
 | 
						|
                     : ossl_time_subtract(b, a);
 | 
						|
}
 | 
						|
 | 
						|
static ossl_unused ossl_inline
 | 
						|
OSSL_TIME ossl_time_multiply(OSSL_TIME a, uint64_t b)
 | 
						|
{
 | 
						|
    OSSL_TIME r;
 | 
						|
    int err = 0;
 | 
						|
 | 
						|
    r.t = safe_mul_time(a.t, b, &err);
 | 
						|
    return err ? ossl_time_infinite() : r;
 | 
						|
}
 | 
						|
 | 
						|
static ossl_unused ossl_inline
 | 
						|
OSSL_TIME ossl_time_divide(OSSL_TIME a, uint64_t b)
 | 
						|
{
 | 
						|
    OSSL_TIME r;
 | 
						|
    int err = 0;
 | 
						|
 | 
						|
    r.t = safe_div_time(a.t, b, &err);
 | 
						|
    return err ? ossl_time_zero() : r;
 | 
						|
}
 | 
						|
 | 
						|
static ossl_unused ossl_inline
 | 
						|
OSSL_TIME ossl_time_muldiv(OSSL_TIME a, uint64_t b, uint64_t c)
 | 
						|
{
 | 
						|
    OSSL_TIME r;
 | 
						|
    int err = 0;
 | 
						|
 | 
						|
    r.t = safe_muldiv_time(a.t, b, c, &err);
 | 
						|
    return err ? ossl_time_zero() : r;
 | 
						|
}
 | 
						|
 | 
						|
/* Return higher of the two given time values. */
 | 
						|
static ossl_unused ossl_inline
 | 
						|
OSSL_TIME ossl_time_max(OSSL_TIME a, OSSL_TIME b)
 | 
						|
{
 | 
						|
    return a.t > b.t ? a : b;
 | 
						|
}
 | 
						|
 | 
						|
/* Return the lower of the two given time values. */
 | 
						|
static ossl_unused ossl_inline
 | 
						|
OSSL_TIME ossl_time_min(OSSL_TIME a, OSSL_TIME b)
 | 
						|
{
 | 
						|
    return a.t < b.t ? a : b;
 | 
						|
}
 | 
						|
 | 
						|
#endif
 |