PPI: Clean up time interface
[senf.git] / Scheduler / ClockService.cci
1 // $Id$
2 //
3 // Copyright (C) 2007 
4 // Fraunhofer Institut fuer offene Kommunikationssysteme (FOKUS)
5 // Kompetenzzentrum fuer Satelitenkommunikation (SatCom)
6 //     Stefan Bund <g0dil@berlios.de>
7 //
8 // This program is free software; you can redistribute it and/or modify
9 // it under the terms of the GNU General Public License as published by
10 // the Free Software Foundation; either version 2 of the License, or
11 // (at your option) any later version.
12 //
13 // This program is distributed in the hope that it will be useful,
14 // but WITHOUT ANY WARRANTY; without even the implied warranty of
15 // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
16 // GNU General Public License for more details.
17 //
18 // You should have received a copy of the GNU General Public License
19 // along with this program; if not, write to the
20 // Free Software Foundation, Inc.,
21 // 59 Temple Place - Suite 330, Boston, MA  02111-1307, USA.
22
23 /** \file
24     \brief ClockService inline non-template implementation */
25
26 // Custom includes
27 #include "boost/date_time/posix_time/posix_time_types.hpp"
28
29 #define prefix_ inline
30 ///////////////////////////////cci.p///////////////////////////////////////
31
32 ///////////////////////////////////////////////////////////////////////////
33 // senf::ClockService
34
35 prefix_ senf::ClockService::clock_type senf::ClockService::now()
36 {
37     // We must make sure to call instance() before fetching the current time since the first call to
38     // instance() will construct the singleton and initialize heartbeat_. If this happens *after*
39     // fetching the current time checkSkew() will detect clock skew since heartbeat_ will be <
40     // current time.
41     return instance().now_i();
42 }
43
44 prefix_ senf::ClockService::abstime_type senf::ClockService::abstime(clock_type clock)
45 {
46 #ifdef BOOST_DATE_TIME_POSIX_TIME_STD_CONFIG
47     return instance().base_ + boost::posix_time::nanoseconds(clock);
48 #else
49     return instance().base_ + boost::posix_time::microseconds((clock+500)/1000);
50 #endif
51 }
52
53 prefix_ senf::ClockService::clock_type senf::ClockService::clock(abstime_type time)
54 {
55     boost::posix_time::time_duration delta (time - instance().base_);
56     return clock_type( delta.ticks() )
57         * clock_type( 1000000000UL / boost::posix_time::time_duration::ticks_per_second() );
58 }
59
60 prefix_ senf::ClockService::clock_type senf::ClockService::nanoseconds(clock_type v)
61 {
62     return v;
63 }
64
65 prefix_ senf::ClockService::clock_type senf::ClockService::microseconds(clock_type v)
66 {
67     return nanoseconds(1000*v);
68 }
69
70 prefix_ senf::ClockService::clock_type senf::ClockService::milliseconds(clock_type v)
71 {
72     return microseconds(1000*v);
73 }
74
75 prefix_ senf::ClockService::clock_type senf::ClockService::seconds(clock_type v)
76 {
77     return milliseconds(1000*v);
78 }
79
80 prefix_ senf::ClockService::clock_type senf::ClockService::minutes(clock_type v)
81 {
82     return seconds(60*v);
83 }
84
85 prefix_ senf::ClockService::clock_type senf::ClockService::hours(clock_type v)
86 {
87     return minutes(60*v);
88 }
89
90 prefix_ senf::ClockService::clock_type senf::ClockService::days(clock_type v)
91 {
92     return hours(24*v);
93 }
94
95 prefix_ void senf::ClockService::restart()
96 {
97     instance().restart_i();
98 }
99
100 ////////////////////////////////////////
101 // private members
102
103 prefix_ senf::ClockService & senf::ClockService::instance()
104 {
105     static ClockService instance;
106     return instance;
107 }
108
109 prefix_ senf::ClockService::clock_type senf::ClockService::now_i()
110 {
111     boost::posix_time::ptime time (boost::posix_time::microsec_clock::universal_time());
112     if (checkSkew(time)) 
113         updateSkew(time);
114     return clock(time);
115 }
116
117 prefix_ bool senf::ClockService::checkSkew(boost::posix_time::ptime time)
118 {
119     return time < heartbeat_ || (time - heartbeat_) > boost::posix_time::seconds(2*CheckInterval);
120 }
121
122 prefix_ void senf::ClockService::clockSkew(boost::posix_time::ptime time,
123                                            boost::posix_time::ptime expected)
124 {
125     base_ += (time - expected);
126 }
127
128 ///////////////////////////////cci.e///////////////////////////////////////
129 #undef prefix_
130
131 \f
132 // Local Variables:
133 // mode: c++
134 // fill-column: 100
135 // comment-column: 40
136 // c-file-style: "senf"
137 // indent-tabs-mode: nil
138 // ispell-local-dictionary: "american"
139 // compile-command: "scons -u test"
140 // End: