FD.io VPP  v17.01.1-3-gc6833f8
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  /* First try /sys version. */
152  if (cpu_freq != 0)
153  return cpu_freq;
154 
155  /* Next try /proc version. */
157  if (cpu_freq != 0)
158  return cpu_freq;
159 
160  /* If /proc/cpuinfo fails (e.g. not running on Linux) fall back to
161  gettimeofday based estimated clock frequency. */
162  return estimate_clock_frequency (1e-3);
163 }
164 
165 #endif /* CLIB_UNIX */
166 
167 /* Initialize time. */
168 void
170 {
171  memset (c, 0, sizeof (c[0]));
175 
176  /* Initially verify frequency every sec */
178 
182 }
183 
184 void
186 {
187  f64 now_reference = unix_time_now ();
188  f64 dtr = now_reference - c->last_verify_reference_time;
189  f64 dtr_max;
190  u64 dtc = c->last_cpu_time - c->last_verify_cpu_time;
191  f64 round_units = 100e5;
192 
194  c->last_verify_reference_time = now_reference;
195 
196  /*
197  * Is the reported reference interval non-positive,
198  * or off by a factor of two - or 8 seconds - whichever is larger?
199  * Someone reset the clock behind our back.
200  */
201  dtr_max = (f64) (2ULL << c->log2_clocks_per_frequency_verify) /
202  (f64) (1ULL << c->log2_clocks_per_second);
203  dtr_max = dtr_max > 8.0 ? dtr_max : 8.0;
204 
205  if (dtr <= 0.0 || dtr > dtr_max)
206  {
208  return;
209  }
210 
211  c->clocks_per_second =
212  flt_round_nearest ((f64) dtc / (dtr * round_units)) * round_units;
214 
215  /* Double time between verifies; max at 64 secs ~ 1 minute. */
218 }
219 
220 /*
221  * fd.io coding-style-patch-verification: ON
222  *
223  * Local Variables:
224  * eval: (c-set-style "gnu")
225  * End:
226  */
uword unformat(unformat_input_t *i, char *fmt,...)
Definition: unformat.c:966
#define UNFORMAT_END_OF_INPUT
Definition: format.h:143
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 f64 unix_time_now(void)
Definition: time.h:227
static void unformat_free(unformat_input_t *i)
Definition: format.h:161
unsigned long u64
Definition: types.h:89
static u64 min_log2_u64(u64 x)
Definition: clib.h:231
static void unformat_skip_line(unformat_input_t *i)
Definition: format.h:220
u32 log2_clocks_per_frequency_verify
Definition: time.h:67
f64 seconds_per_clock
Definition: time.h:57
void clib_time_init(clib_time_t *c)
Definition: time.c:169
void clib_time_verify_frequency(clib_time_t *c)
Definition: time.c:185
svmdb_client_t * c
void unformat_init_unix_file(unformat_input_t *input, int file_descriptor)
Definition: unformat.c:1052
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:308
double f64
Definition: types.h:142
static f64 estimate_clock_frequency(f64 sample_time)
Definition: time.c:52
static uword unformat_check_input(unformat_input_t *i)
Definition: format.h:169
u64 init_cpu_time
Definition: time.h:60
struct _unformat_input_t unformat_input_t