| Patrick Ohly | a75244c | 2009-02-12 05:03:35 +0000 | [diff] [blame] | 1 | /* | 
 | 2 |  * Utility code which helps transforming between two different time | 
 | 3 |  * bases, called "source" and "target" time in this code. | 
 | 4 |  * | 
 | 5 |  * Source time has to be provided via the timecounter API while target | 
 | 6 |  * time is accessed via a function callback whose prototype | 
 | 7 |  * intentionally matches ktime_get() and ktime_get_real(). These | 
 | 8 |  * interfaces where chosen like this so that the code serves its | 
 | 9 |  * initial purpose without additional glue code. | 
 | 10 |  * | 
 | 11 |  * This purpose is synchronizing a hardware clock in a NIC with system | 
 | 12 |  * time, in order to implement the Precision Time Protocol (PTP, | 
 | 13 |  * IEEE1588) with more accurate hardware assisted time stamping.  In | 
 | 14 |  * that context only synchronization against system time (= | 
 | 15 |  * ktime_get_real()) is currently needed. But this utility code might | 
 | 16 |  * become useful in other situations, which is why it was written as | 
 | 17 |  * general purpose utility code. | 
 | 18 |  * | 
 | 19 |  * The source timecounter is assumed to return monotonically | 
 | 20 |  * increasing time (but this code does its best to compensate if that | 
 | 21 |  * is not the case) whereas target time may jump. | 
 | 22 |  * | 
 | 23 |  * The target time corresponding to a source time is determined by | 
 | 24 |  * reading target time, reading source time, reading target time | 
 | 25 |  * again, then assuming that average target time corresponds to source | 
 | 26 |  * time. In other words, the assumption is that reading the source | 
 | 27 |  * time is slow and involves equal time for sending the request and | 
 | 28 |  * receiving the reply, whereas reading target time is assumed to be | 
 | 29 |  * fast. | 
 | 30 |  * | 
 | 31 |  * Copyright (C) 2009 Intel Corporation. | 
 | 32 |  * Author: Patrick Ohly <patrick.ohly@intel.com> | 
 | 33 |  * | 
 | 34 |  * This program is free software; you can redistribute it and/or modify it | 
 | 35 |  * under the terms and conditions of the GNU General Public License, | 
 | 36 |  * version 2, as published by the Free Software Foundation. | 
 | 37 |  * | 
 | 38 |  * This program is distributed in the hope it will be useful, but WITHOUT | 
 | 39 |  * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or | 
 | 40 |  * FITNESS FOR A PARTICULAR PURPOSE. * See the GNU General Public License for | 
 | 41 |  * more details. | 
 | 42 |  * | 
 | 43 |  * You should have received a copy of the GNU General Public License along with | 
 | 44 |  * this program; if not, write to the Free Software Foundation, Inc., | 
 | 45 |  * 51 Franklin St - Fifth Floor, Boston, MA 02110-1301 USA. | 
 | 46 |  */ | 
 | 47 | #ifndef _LINUX_TIMECOMPARE_H | 
 | 48 | #define _LINUX_TIMECOMPARE_H | 
 | 49 |  | 
 | 50 | #include <linux/clocksource.h> | 
 | 51 | #include <linux/ktime.h> | 
 | 52 |  | 
 | 53 | /** | 
 | 54 |  * struct timecompare - stores state and configuration for the two clocks | 
 | 55 |  * | 
 | 56 |  * Initialize to zero, then set source/target/num_samples. | 
 | 57 |  * | 
 | 58 |  * Transformation between source time and target time is done with: | 
 | 59 |  * target_time = source_time + offset + | 
 | 60 |  *               (source_time - last_update) * skew / | 
 | 61 |  *               TIMECOMPARE_SKEW_RESOLUTION | 
 | 62 |  * | 
 | 63 |  * @source:          used to get source time stamps via timecounter_read() | 
 | 64 |  * @target:          function returning target time (for example, ktime_get | 
 | 65 |  *                   for monotonic time, or ktime_get_real for wall clock) | 
 | 66 |  * @num_samples:     number of times that source time and target time are to | 
 | 67 |  *                   be compared when determining their offset | 
 | 68 |  * @offset:          (target time - source time) at the time of the last update | 
 | 69 |  * @skew:            average (target time - source time) / delta source time * | 
 | 70 |  *                   TIMECOMPARE_SKEW_RESOLUTION | 
 | 71 |  * @last_update:     last source time stamp when time offset was measured | 
 | 72 |  */ | 
 | 73 | struct timecompare { | 
 | 74 | 	struct timecounter *source; | 
 | 75 | 	ktime_t (*target)(void); | 
 | 76 | 	int num_samples; | 
 | 77 |  | 
 | 78 | 	s64 offset; | 
 | 79 | 	s64 skew; | 
 | 80 | 	u64 last_update; | 
 | 81 | }; | 
 | 82 |  | 
 | 83 | /** | 
 | 84 |  * timecompare_transform - transform source time stamp into target time base | 
 | 85 |  * @sync:            context for time sync | 
 | 86 |  * @source_tstamp:   the result of timecounter_read() or | 
 | 87 |  *                   timecounter_cyc2time() | 
 | 88 |  */ | 
 | 89 | extern ktime_t timecompare_transform(struct timecompare *sync, | 
 | 90 | 				     u64 source_tstamp); | 
 | 91 |  | 
 | 92 | /** | 
 | 93 |  * timecompare_offset - measure current (target time - source time) offset | 
 | 94 |  * @sync:            context for time sync | 
 | 95 |  * @offset:          average offset during sample period returned here | 
 | 96 |  * @source_tstamp:   average source time during sample period returned here | 
 | 97 |  * | 
 | 98 |  * Returns number of samples used. Might be zero (= no result) in the | 
 | 99 |  * unlikely case that target time was monotonically decreasing for all | 
 | 100 |  * samples (= broken). | 
 | 101 |  */ | 
 | 102 | extern int timecompare_offset(struct timecompare *sync, | 
 | 103 | 			      s64 *offset, | 
 | 104 | 			      u64 *source_tstamp); | 
 | 105 |  | 
 | 106 | extern void __timecompare_update(struct timecompare *sync, | 
 | 107 | 				 u64 source_tstamp); | 
 | 108 |  | 
 | 109 | /** | 
 | 110 |  * timecompare_update - update offset and skew by measuring current offset | 
 | 111 |  * @sync:            context for time sync | 
 | 112 |  * @source_tstamp:   the result of timecounter_read() or | 
 | 113 |  *                   timecounter_cyc2time(), pass zero to force update | 
 | 114 |  * | 
 | 115 |  * Updates are only done at most once per second. | 
 | 116 |  */ | 
 | 117 | static inline void timecompare_update(struct timecompare *sync, | 
 | 118 | 				      u64 source_tstamp) | 
 | 119 | { | 
 | 120 | 	if (!source_tstamp || | 
 | 121 | 	    (s64)(source_tstamp - sync->last_update) >= NSEC_PER_SEC) | 
 | 122 | 		__timecompare_update(sync, source_tstamp); | 
 | 123 | } | 
 | 124 |  | 
 | 125 | #endif /* _LINUX_TIMECOMPARE_H */ |