/* * Mach Operating System * Copyright (c) 1991,1990,1989,1988,1987 Carnegie Mellon University * All Rights Reserved. * * Permission to use, copy, modify and distribute this software and its * documentation is hereby granted, provided that both the copyright * notice and this permission notice appear in all copies of the * software, derivative works or modified versions, and any portions * thereof, and that both notices appear in supporting documentation. * * CARNEGIE MELLON ALLOWS FREE USE OF THIS SOFTWARE IN ITS "AS IS" * CONDITION. CARNEGIE MELLON DISCLAIMS ANY LIABILITY OF ANY KIND FOR * ANY DAMAGES WHATSOEVER RESULTING FROM THE USE OF THIS SOFTWARE. * * Carnegie Mellon requests users of this software to return to * * Software Distribution Coordinator or Software.Distribution@CS.CMU.EDU * School of Computer Science * Carnegie Mellon University * Pittsburgh PA 15213-3890 * * any improvements or extensions that they make and grant Carnegie Mellon * the rights to redistribute these changes. */ #ifndef _MACH_TIME_VALUE_H_ #define _MACH_TIME_VALUE_H_ #include /* * Time value returned by kernel. */ struct rpc_time_value { /* TODO: this should be 64 bits regardless of the arch to be Y2038 proof. */ rpc_long_integer_t seconds; integer_t microseconds; }; /* * Time value used by kernel interfaces. Ideally they should be migrated * to use time_value64 below. */ struct time_value { long_integer_t seconds; integer_t microseconds; }; typedef struct time_value time_value_t; #ifdef KERNEL typedef struct rpc_time_value rpc_time_value_t; #else typedef struct time_value rpc_time_value_t; #endif /* * Time value used internally by the kernel that uses 64 bits to track seconds * and nanoseconds. Note that the current resolution is only microseconds. */ struct time_value64 { int64_t seconds; int64_t nanoseconds; }; typedef struct time_value64 time_value64_t; /** * Functions used by Mig to perform user to kernel conversion and vice-versa. * We only do this because we may run a 64 bit kernel with a 32 bit user space. */ static __inline__ rpc_time_value_t convert_time_value_to_user(time_value_t tv) { rpc_time_value_t user = {.seconds = tv.seconds, .microseconds = tv.microseconds}; return user; } static __inline__ time_value_t convert_time_value_from_user(rpc_time_value_t tv) { time_value_t kernel = {.seconds = tv.seconds, .microseconds = tv.microseconds}; return kernel; } /* * Macros to manipulate time values. Assume that time values * are normalized (microseconds <= 999999). */ #define TIME_MICROS_MAX (1000000) #define TIME_NANOS_MAX (1000000000) #define time_value_assert(val) \ assert(0 <= (val)->microseconds && (val)->microseconds < TIME_MICROS_MAX); #define time_value64_assert(val) \ assert(0 <= (val)->nanoseconds && (val)->nanoseconds < TIME_NANOS_MAX); #define time_value_add_usec(val, micros) { \ time_value_assert(val); \ if (((val)->microseconds += (micros)) \ >= TIME_MICROS_MAX) { \ (val)->microseconds -= TIME_MICROS_MAX; \ (val)->seconds++; \ } \ time_value_assert(val); \ } #define time_value64_add_nanos(val, nanos) { \ time_value64_assert(val); \ if (((val)->nanoseconds += (nanos)) \ >= TIME_NANOS_MAX) { \ (val)->nanoseconds -= TIME_NANOS_MAX; \ (val)->seconds++; \ } \ time_value64_assert(val); \ } #define time_value64_sub_nanos(val, nanos) { \ time_value64_assert(val); \ if (((val)->nanoseconds -= (nanos)) < 0) { \ (val)->nanoseconds += TIME_NANOS_MAX; \ (val)->seconds--; \ } \ time_value64_assert(val); \ } #define time_value_add(result, addend) { \ time_value_assert(addend); \ (result)->seconds += (addend)->seconds; \ time_value_add_usec(result, (addend)->microseconds); \ } #define time_value64_add(result, addend) { \ time_value64_assert(addend); \ (result)->seconds += (addend)->seconds; \ time_value64_add_nanos(result, (addend)->nanoseconds); \ } #define time_value64_sub(result, subtrahend) { \ time_value64_assert(subtrahend); \ (result)->seconds -= (subtrahend)->seconds; \ time_value64_sub_nanos(result, (subtrahend)->nanoseconds); \ } #define time_value64_init(tv) { \ (tv)->seconds = 0; \ (tv)->nanoseconds = 0; \ } #define TIME_VALUE64_TO_TIME_VALUE(tv64, tv) do { \ (tv)->seconds = (tv64)->seconds; \ (tv)->microseconds = (tv64)->nanoseconds / 1000; \ } while(0) #define TIME_VALUE_TO_TIME_VALUE64(tv, tv64) do { \ (tv64)->seconds = (tv)->seconds; \ (tv64)->nanoseconds = (tv)->microseconds * 1000; \ } while(0) /* * Time value available through the mapped-time interface. * Read this mapped value with * do { * secs = mtime->seconds; * __sync_synchronize(); * usecs = mtime->microseconds; * __sync_synchronize(); * } while (secs != mtime->check_seconds); */ typedef struct mapped_time_value { integer_t seconds; integer_t microseconds; integer_t check_seconds; struct time_value64 time_value; int64_t check_seconds64; struct time_value64 uptime_value; int64_t check_upseconds64; } mapped_time_value_t; /* Macros for converting between struct timespec and time_value_t. */ #define TIME_VALUE_TO_TIMESPEC(tv, ts) do { \ (ts)->tv_sec = (tv)->seconds; \ (ts)->tv_nsec = (tv)->microseconds * 1000; \ } while(0) #define TIMESPEC_TO_TIME_VALUE(tv, ts) do { \ (tv)->seconds = (ts)->tv_sec; \ (tv)->microseconds = (ts)->tv_nsec / 1000; \ } while(0) /* Macros for converting between struct timespec and time_value64_t. */ #define TIME_VALUE64_TO_TIMESPEC(tv, ts) do { \ (ts)->tv_sec = (tv)->seconds; \ (ts)->tv_nsec = (tv)->nanoseconds; \ } while(0) #define TIMESPEC_TO_TIME_VALUE64(tv, ts) do { \ (tv)->seconds = (ts)->tv_sec; \ (tv)->nanoseconds = (ts)->tv_nsec; \ } while(0) #endif /* _MACH_TIME_VALUE_H_ */