FD.io VPP  v17.07-30-g839fa73
Vector Packet Processing
time.c
Go to the documentation of this file.
1 /*
2  * Copyright (c) 2015 Cisco and/or its affiliates.
3  * Licensed under the Apache License, Version 2.0 (the "License");
4  * you may not use this file except in compliance with the License.
5  * You may obtain a copy of the License at:
6  *
7  * http://www.apache.org/licenses/LICENSE-2.0
8  *
9  * Unless required by applicable law or agreed to in writing, software
10  * distributed under the License is distributed on an "AS IS" BASIS,
11  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12  * See the License for the specific language governing permissions and
13  * limitations under the License.
14  */
15 /*
16  Copyright (c) 2005 Eliot Dresselhaus
17 
18  Permission is hereby granted, free of charge, to any person obtaining
19  a copy of this software and associated documentation files (the
20  "Software"), to deal in the Software without restriction, including
21  without limitation the rights to use, copy, modify, merge, publish,
22  distribute, sublicense, and/or sell copies of the Software, and to
23  permit persons to whom the Software is furnished to do so, subject to
24  the following conditions:
25 
26  The above copyright notice and this permission notice shall be
27  included in all copies or substantial portions of the Software.
28 
29  THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
30  EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
31  MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
32  NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
33  LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
34  OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
35  WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
36 */
37 
38 #include <vppinfra/os.h>
39 #include <vppinfra/time.h>
40 #include <vppinfra/format.h>
41 #include <vppinfra/cpu.h>
42 
43 #ifdef CLIB_UNIX
44 
45 #include <math.h>
46 #include <sys/time.h>
47 #include <fcntl.h>
48 
49 /* Not very accurate way of determining cpu clock frequency
50  for unix. Better to use /proc/cpuinfo on linux. */
51 static f64
53 {
54  /* Round to nearest 100KHz. */
55  const f64 round_to_units = 100e5;
56 
57  f64 time_now, time_start, time_limit, freq;
58  u64 ifreq, t[2];
59 
60  time_start = time_now = unix_time_now ();
61  time_limit = time_now + sample_time;
62  t[0] = clib_cpu_time_now ();
63  while (time_now < time_limit)
64  time_now = unix_time_now ();
65  t[1] = clib_cpu_time_now ();
66 
67  freq = (t[1] - t[0]) / (time_now - time_start);
68  ifreq = flt_round_nearest (freq / round_to_units);
69  freq = ifreq * round_to_units;
70 
71  return freq;
72 }
73 
74 /* Fetch cpu frequency via parseing /proc/cpuinfo.
75  Only works for Linux. */
76 static f64
78 {
79  f64 cpu_freq = 1e9; /* better than 40... */
80  f64 ppc_timebase = 0; /* warnings be gone */
81  int fd;
82  unformat_input_t input;
83 
84 /* $$$$ aarch64 kernel doesn't report "cpu MHz" */
85 #if defined(__aarch64__)
86  return 0.0;
87 #endif
88 
89  cpu_freq = 0;
90  fd = open ("/proc/cpuinfo", 0);
91  if (fd < 0)
92  return cpu_freq;
93 
94  unformat_init_unix_file (&input, fd);
95 
96  ppc_timebase = 0;
98  {
99  if (unformat (&input, "cpu MHz : %f", &cpu_freq))
100  cpu_freq *= 1e6;
101  else if (unformat (&input, "timebase : %f", &ppc_timebase))
102  ;
103  else
104  unformat_skip_line (&input);
105  }
106 
107  unformat_free (&input);
108 
109  close (fd);
110 
111  /* Override CPU frequency with time base for PPC. */
112  if (ppc_timebase != 0)
113  cpu_freq = ppc_timebase;
114 
115  return cpu_freq;
116 }
117 
118 /* Fetch cpu frequency via reading /sys/devices/system/cpu/cpu0/cpufreq/cpuinfo_max_freq
119  Only works for Linux. */
120 static f64
122 {
123  f64 cpu_freq;
124  int fd;
125  unformat_input_t input;
126 
127  /* Time stamp always runs at max frequency. */
128  cpu_freq = 0;
129  fd = open ("/sys/devices/system/cpu/cpu0/cpufreq/cpuinfo_max_freq", 0);
130  if (fd < 0)
131  goto done;
132 
133  unformat_init_unix_file (&input, fd);
134  unformat (&input, "%f", &cpu_freq);
135  cpu_freq *= 1e3; /* measured in kHz */
136  unformat_free (&input);
137  close (fd);
138 done:
139  return cpu_freq;
140 }
141 
142 f64
144 {
145  f64 cpu_freq;
146 
147  if (clib_cpu_supports_invariant_tsc ())
148  return estimate_clock_frequency (1e-3);
149 
150 #if defined (__aarch64__)
151  u64 tsc;
152  asm volatile ("mrs %0, CNTFRQ_EL0":"=r" (tsc));
153  return (f64) tsc;
154 #endif
155 
156  /* First try /sys version. */
158  if (cpu_freq != 0)
159  return cpu_freq;
160 
161  /* Next try /proc version. */
163  if (cpu_freq != 0)
164  return cpu_freq;
165 
166  /* If /proc/cpuinfo fails (e.g. not running on Linux) fall back to
167  gettimeofday based estimated clock frequency. */
168  return estimate_clock_frequency (1e-3);
169 }
170 
171 #endif /* CLIB_UNIX */
172 
173 /* Initialize time. */
174 void
176 {
177  memset (c, 0, sizeof (c[0]));
181 
182  /* Initially verify frequency every sec */
184 
188 }
189 
190 void
192 {
193  f64 now_reference = unix_time_now ();
194  f64 dtr = now_reference - c->last_verify_reference_time;
195  f64 dtr_max;
196  u64 dtc = c->last_cpu_time - c->last_verify_cpu_time;
197  f64 round_units = 100e5;
198 
200  c->last_verify_reference_time = now_reference;
201 
202  /*
203  * Is the reported reference interval non-positive,
204  * or off by a factor of two - or 8 seconds - whichever is larger?
205  * Someone reset the clock behind our back.
206  */
207  dtr_max = (f64) (2ULL << c->log2_clocks_per_frequency_verify) /
208  (f64) (1ULL << c->log2_clocks_per_second);
209  dtr_max = dtr_max > 8.0 ? dtr_max : 8.0;
210 
211  if (dtr <= 0.0 || dtr > dtr_max)
212  {
214  return;
215  }
216 
217  c->clocks_per_second =
218  flt_round_nearest ((f64) dtc / (dtr * round_units)) * round_units;
220 
221  /* Double time between verifies; max at 64 secs ~ 1 minute. */
224 }
225 
226 /*
227  * fd.io coding-style-patch-verification: ON
228  *
229  * Local Variables:
230  * eval: (c-set-style "gnu")
231  * End:
232  */
f64 clocks_per_second
Definition: time.h:53
static u64 clib_cpu_time_now(void)
Definition: time.h:73
u32 log2_clocks_per_second
Definition: time.h:67
u64 last_verify_cpu_time
Definition: time.h:62
static void unformat_skip_line(unformat_input_t *i)
Definition: format.h:220
static f64 unix_time_now(void)
Definition: time.h:227
unsigned long u64
Definition: types.h:89
static u64 min_log2_u64(u64 x)
Definition: clib.h:237
u32 log2_clocks_per_frequency_verify
Definition: time.h:67
struct _unformat_input_t unformat_input_t
f64 seconds_per_clock
Definition: time.h:57
void clib_time_init(clib_time_t *c)
Definition: time.c:175
void clib_time_verify_frequency(clib_time_t *c)
Definition: time.c:191
#define UNFORMAT_END_OF_INPUT
Definition: format.h:143
svmdb_client_t * c
static f64 clock_frequency_from_sys_filesystem(void)
Definition: time.c:121
u64 last_cpu_time
Definition: time.h:50
f64 os_cpu_clock_frequency(void)
Definition: time.c:143
static f64 clock_frequency_from_proc_filesystem(void)
Definition: time.c:77
f64 last_verify_reference_time
Definition: time.h:65
static word flt_round_nearest(f64 x)
Definition: clib.h:314
double f64
Definition: types.h:142
static f64 estimate_clock_frequency(f64 sample_time)
Definition: time.c:52
static void unformat_free(unformat_input_t *i)
Definition: format.h:161
u64 init_cpu_time
Definition: time.h:60
void unformat_init_unix_file(unformat_input_t *input, int file_descriptor)
Definition: unformat.c:1058
uword unformat(unformat_input_t *i, const char *fmt,...)
Definition: unformat.c:972
static uword unformat_check_input(unformat_input_t *i)
Definition: format.h:169