ProcessWatcher.cpp
Go to the documentation of this file.
1 /*
2 * This file is part of ArmarX.
3 *
4 * Copyright (C) 2011-2016, High Performance Humanoid Technologies (H2T), Karlsruhe Institute of Technology (KIT), all rights reserved.
5 *
6 * ArmarX is free software; you can redistribute it and/or modify
7 * it under the terms of the GNU General Public License version 2 as
8 * published by the Free Software Foundation.
9 *
10 * ArmarX is distributed in the hope that it will be useful, but
11 * WITHOUT ANY WARRANTY; without even the implied warranty of
12 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
13 * GNU General Public License for more details.
14 *
15 * You should have received a copy of the GNU General Public License
16 * along with this program. If not, see <http://www.gnu.org/licenses/>.
17 *
18 * @package ArmarX::
19 * @author Mirko Waechter ( mirko.waechter at kit dot edu)
20 * @date 2013
21 * @copyright http://www.gnu.org/licenses/gpl-2.0.txt
22 * GNU General Public License
23 */
24 
25 #include "ProcessWatcher.h"
26 
27 #include <ctype.h> // for isspace
28 
29 #include <fstream>
30 #include <memory> // for allocator, operator==
31 #include <sstream> // for operator<<, ifstream, etc
32 #include <unordered_set>
33 
34 #include <sys/utsname.h> // for uname, utsname
35 #include <unistd.h> // for sysconf, _SC_CLK_TCK
36 
37 #include <IceUtil/Handle.h> // for HandleBase
38 
39 #include <SimoxUtility/algorithm/string/string_tools.h>
40 
41 #include "ArmarXCore/core/logging/LogSender.h" // for LogSender
42 #include "ArmarXCore/core/logging/Logging.h" // for ARMARX_WARNING_S
45 #include "ArmarXCore/interface/core/ThreadingIceBase.h" // for upCast
46 
47 #include "../exceptions/local/ExpressionException.h"
48 #include <ext/alloc_traits.h>
49 #include <malloc.h> // for mallinfo
50 
51 namespace armarx
52 {
54  {
55  long pid = LogSender::getProcessId();
56  Hertz = sysconf(_SC_CLK_TCK);
57  std::stringstream ss;
58  ss << "/proc/" << pid << "/stat";
59  processCpuDataFilename = ss.str();
60  processCpuUsage.processId = pid;
61  cpuUsageFromProcFile();
62  }
63 
64  void
66  {
67  if (!watcherTask)
68  {
69  watcherTask =
70  new PeriodicTask<ProcessWatcher>(this, &ProcessWatcher::watch, 300, false);
71  }
72 
73  watcherTask->start();
74  }
75 
76  void
78  {
79  watcherTask->stop();
80  watcherTask = nullptr;
81  }
82 
83  void
84  ProcessWatcher::addThread(int processId, int threadId)
85  {
87  std::unique_lock lock(threadWatchListMutex);
88  ThreadUsage usage;
89  usage.processId = processId;
90  usage.threadId = threadId;
91  usage.lastJiffies = GetThreadJiffies(processId, threadId);
92  usage.lastUpdate = IceUtil::Time::now();
93  usage.load = 0;
94  threadWatchList.insert(usage);
95  }
96 
97  void
99  {
100  addThread(LogSender::getProcessId(), threadId);
101  }
102 
103  void
104  ProcessWatcher::watch()
105  {
106  ARMARX_TRACE;
107  runThreadWatchList();
108  cpuUsageFromProcFile();
109  getMemoryUsage();
110  reportCpuUsage();
112  }
113 
114  void
115  ProcessWatcher::runThreadWatchList()
116  {
117  ARMARX_TRACE;
118  std::unique_lock lock(threadWatchListMutex);
119  std::unordered_set<ThreadUsage> newset;
120  std::unordered_set<ThreadUsage>::iterator it = threadWatchList.begin();
121 
122  for (; it != threadWatchList.end(); it++)
123  {
124  ARMARX_TRACE;
125  ThreadUsage usage = *it;
126  IceUtil::Time queryTime = IceUtil::Time::now();
127  int curJiffies = GetThreadJiffies(usage.processId, usage.threadId);
128 
129  if (GetHertz() != -1)
130  {
131  usage.load = (curJiffies - usage.lastJiffies) /
132  ((queryTime - usage.lastUpdate).toSecondsDouble() * GetHertz());
133  }
134  else
135  {
136  usage.load = 0;
137  }
138 
139  usage.lastJiffies = curJiffies;
140  usage.lastUpdate = queryTime;
141  newset.insert(usage);
142  }
143 
144  threadWatchList.swap(newset);
145  }
146 
147 
148 #if !defined(_WIN32) && !defined(__APPLE__)
149  int
151  {
152  ARMARX_TRACE;
153  static std::mutex mutex;
154  std::unique_lock lock(mutex);
155  static int value = -2;
156 
157  if (value == -2) // init value
158  {
159  ARMARX_TRACE;
160  static std::string _release;
161 
162  if (_release.empty())
163  {
164  struct utsname utsinfo;
165  uname(&utsinfo);
166  _release = utsinfo.release;
167  }
168 
169  std::string fileName = "/boot/config-" + _release;
170  std::ifstream configFile(fileName.c_str());
171 
172  ARMARX_TRACE;
173  // procFile.open(fileName, ios::out);
174  if (!configFile.is_open())
175  {
176  ARMARX_WARNING_S << "Cannot open " << fileName
177  << " for reading the cpu hertz value";
178  value = -1;
179  }
180  else
181  {
182  value = -1;
183  std::string line;
184  const std::string searchString = "CONFIG_HZ=";
185 
186  while (getline(configFile, line))
187  {
188  ARMARX_TRACE;
189  std::size_t pos = line.find(searchString);
190 
191  if (pos != std::string::npos)
192  {
193  std::string hzString = line.substr(pos + searchString.length());
194  value = std::stoi(hzString);
195  break;
196  }
197  }
198  }
199  }
200 
201  return value;
202  }
203 
204  void
205  ProcessWatcher::cpuUsageFromProcFile()
206  {
207  ARMARX_TRACE;
208  std::string line;
209 
210  std::ifstream file(processCpuDataFilename);
211 
212  if (!file.is_open())
213  {
214  ARMARX_WARNING_S << "File " << processCpuDataFilename << " does not exist";
215  }
216 
217 
218  std::getline(file, line);
219  file.close();
220 
221  std::vector<std::string> pidElements = simox::alg::split(line, "\t ");
222 
223  int currentUtime = std::stoi(pidElements.at(eUTime));
224  int currentStime = std::stoi(pidElements.at(eSTime));
225 
226  int currentCUtime = std::stoi(pidElements.at(eCUTIME));
227  int currentCStime = std::stoi(pidElements.at(eCSTIME));
228  {
229  ARMARX_TRACE;
230  std::unique_lock guard{processCpuUsageMutex};
231 
232  IceUtil::Time queryTime = IceUtil::Time::now();
233 
234  processCpuUsage.proc_total_time =
235  100 *
236  (double)((currentStime + currentCStime - processCpuUsage.lastStime -
237  processCpuUsage.lastCStime) +
238  (currentUtime + currentCUtime - processCpuUsage.lastCUtime -
239  processCpuUsage.lastUtime)) /
240  ((queryTime - processCpuUsage.lastUpdate).toSecondsDouble() * Hertz);
241 
242  processCpuUsage.lastUpdate = queryTime;
243  processCpuUsage.lastUtime = currentUtime;
244  processCpuUsage.lastStime = currentStime;
245 
246  processCpuUsage.lastCUtime = currentCUtime;
247  processCpuUsage.lastCStime = currentCStime;
248  }
249  }
250 
251  void
252  ProcessWatcher::getMemoryUsage()
253  {
254  ARMARX_TRACE;
255  struct mallinfo _mallinfo;
256 
257  _mallinfo = mallinfo();
258 
259  {
260  std::unique_lock guard{processMemoryUsageMutex};
261  memoryUsage.fastbinBlocks = _mallinfo.fsmblks;
262  memoryUsage.totalAllocatedSpace = _mallinfo.uordblks;
263  memoryUsage.totalFreeSpace = _mallinfo.fordblks;
264  }
265  }
266 
267  std::map<int, int>
268  ProcessWatcher::GetThreadListJiffies(int processId, std::vector<int> threadIds)
269  {
270  ARMARX_TRACE;
271  std::map<int, int> resultMap;
272  for (int threadId : threadIds)
273  {
274  ARMARX_TRACE;
275  resultMap[threadId] = 0;
276  std::stringstream fileName;
277  fileName << "/proc/" << threadId << "/task/" << threadId << "/stat";
278  std::ifstream statFile(fileName.str().c_str());
279 
280  if (!statFile.is_open())
281  {
282  ARMARX_WARNING_S << "Cannot open " << fileName.str()
283  << " for reading the hertz value";
284  resultMap[threadId] = 0;
285  continue;
286  }
287 
288  std::string line;
289 
290  while (getline(statFile, line))
291  {
292  ARMARX_TRACE;
293  simox::alg::trim(line);
294 
295  std::vector<std::string> stringVec = simox::alg::split(line, " ");
296 
297  ARMARX_CHECK_LESS(14, stringVec.size());
298  int userTimeJiffies = atoi(stringVec.at(13).c_str());
299  int kernelTimeJiffies = atoi(stringVec.at(14).c_str());
300 
301  resultMap[threadId] = userTimeJiffies + kernelTimeJiffies;
302  }
303  }
304 
305  return resultMap;
306  }
307 #else
308 
309  int
311  {
312  return -1;
313  }
314 
315  std::map<int, int>
316  ProcessWatcher::GetThreadListJiffies(int processId, std::vector<int> threadIds)
317  {
318  return std::map<int, int>();
319  }
320 
321  void
322  ProcessWatcher::cpuUsageFromProcFile()
323  {
324  }
325 
326  void
327  ProcessWatcher::getMemoryUsage()
328  {
329  }
330 
331 #endif
332 
333  bool
335  {
336  if (processId < rhs.processId)
337  {
338  return true;
339  }
340 
341  if (threadId < rhs.threadId)
342  {
343  return true;
344  }
345 
346  return false;
347  }
348 
349  bool
351  {
352  if (processId == rhs.processId && threadId == rhs.threadId)
353  {
354  return true;
355  }
356 
357  return false;
358  }
359 
360  int
361  ProcessWatcher::GetThreadJiffies(int processId, int threadId)
362  {
363  ARMARX_TRACE;
364  std::map<int, int> result = GetThreadListJiffies(processId, std::vector<int>(1, threadId));
365  std::map<int, int>::iterator it = result.find(threadId);
366 
367  if (it == result.end())
368  {
369  return 0;
370  }
371 
372  return it->second;
373  }
374 
375  void
377  {
378  std::unique_lock guard{processCpuUsageMutex};
379  profiler->logProcessCpuUsage(processCpuUsage.proc_total_time);
380  }
381 
382  void
384  {
385  std::unique_lock guard{processMemoryUsageMutex};
386  profiler->logProcessMemoryUsage(memoryUsage.totalFreeSpace);
387  }
388 
389  double
391  {
392  return getThreadLoad(LogSender::getProcessId(), threadId);
393  }
394 
395  double
396  ProcessWatcher::getThreadLoad(int processId, int threadId)
397  {
398  ARMARX_TRACE;
399  ThreadUsage query;
400  query.processId = processId;
401  query.threadId = threadId;
402  std::unique_lock lock(threadWatchListMutex);
403  auto it = threadWatchList.find(query);
404 
405  ARMARX_TRACE;
406  if (it != threadWatchList.end())
407  {
408  return it->load * 100;
409  }
410  else
411  {
412  return 0.0;
413  }
414  }
415 
418  {
419  std::unique_lock guard{processCpuUsageMutex};
420  return processCpuUsage;
421  }
422 
423  void
424  ProcessWatcher::removeThread(int processId, int threadId)
425  {
426  ARMARX_TRACE;
427  ThreadUsage query;
428  query.processId = processId;
429  query.threadId = threadId;
430  std::unique_lock lock(threadWatchListMutex);
431  threadWatchList.erase(query);
432  }
433 
434  void
436  {
438  }
439 } // namespace armarx
armarx::ProcessWatcher::stop
void stop()
Definition: ProcessWatcher.cpp:77
armarx::CpuUsage::lastCUtime
int lastCUtime
Definition: ProcessWatcher.h:57
armarx::ThreadUsage::lastUpdate
IceUtil::Time lastUpdate
Definition: ProcessWatcher.h:45
ProcessWatcher.h
armarx::CpuUsage
Definition: ProcessWatcher.h:52
armarx::ProcessWatcher::GetThreadJiffies
static int GetThreadJiffies(int processId, int threadId)
Definition: ProcessWatcher.cpp:361
armarx::ProcessWatcher::GetHertz
static int GetHertz()
Definition: ProcessWatcher.cpp:150
armarx::ProcessWatcher::reportMemoryUsage
void reportMemoryUsage()
Definition: ProcessWatcher.cpp:383
armarx::MemoryUsage::fastbinBlocks
int fastbinBlocks
Definition: ProcessWatcher.h:65
armarx::ProcessWatcher::getThreadLoad
double getThreadLoad(int processId, int threadId)
Definition: ProcessWatcher.cpp:396
PeriodicTask.h
ARMARX_CHECK_LESS
#define ARMARX_CHECK_LESS(lhs, rhs)
This macro evaluates whether lhs is less (<) than rhs and if it turns out to be false it will throw a...
Definition: ExpressionException.h:102
armarx::ProcessWatcher::removeThread
void removeThread(int processId, int threadId)
Definition: ProcessWatcher.cpp:424
armarx::ThreadUsage::operator<
bool operator<(const ThreadUsage &rhs) const
Definition: ProcessWatcher.cpp:334
ARMARX_TRACE
#define ARMARX_TRACE
Definition: trace.h:77
armarx::CpuUsage::proc_total_time
double proc_total_time
Definition: ProcessWatcher.h:60
cxxopts::value
std::shared_ptr< Value > value()
Definition: cxxopts.hpp:855
armarx::CpuUsage::lastStime
int lastStime
Definition: ProcessWatcher.h:56
armarx::ProcessWatcher::ProcessWatcher
ProcessWatcher(Profiler::ProfilerPtr profiler)
Definition: ProcessWatcher.cpp:53
armarx::ProcessWatcher::GetThreadListJiffies
static std::map< int, int > GetThreadListJiffies(int processId, std::vector< int > threadIds)
Definition: ProcessWatcher.cpp:268
armarx::ThreadUsage::processId
int processId
Definition: ProcessWatcher.h:44
armarx::ProcessWatcher::addThread
void addThread(int processId, int threadId)
Definition: ProcessWatcher.cpp:84
armarx::ThreadUsage::load
double load
Definition: ProcessWatcher.h:47
ARMARX_WARNING_S
#define ARMARX_WARNING_S
Definition: Logging.h:213
armarx::armem::Time
armarx::core::time::DateTime Time
Definition: forward_declarations.h:13
armarx::ProcessWatcher::getProcessCpuUsage
armarx::CpuUsage getProcessCpuUsage()
Definition: ProcessWatcher.cpp:417
armarx::CpuUsage::lastUtime
int lastUtime
Definition: ProcessWatcher.h:55
armarx::LogSender::getProcessId
static long getProcessId()
Definition: LogSender.cpp:459
armarx::CpuUsage::lastCStime
int lastCStime
Definition: ProcessWatcher.h:58
armarx::ThreadUsage::lastJiffies
int lastJiffies
Definition: ProcessWatcher.h:46
armarx::CpuUsage::processId
long processId
Definition: ProcessWatcher.h:54
Profiler.h
armarx::ThreadUsage::threadId
int threadId
Definition: ProcessWatcher.h:43
armarx::MemoryUsage::totalAllocatedSpace
int totalAllocatedSpace
Definition: ProcessWatcher.h:66
LogSender.h
armarx::Profiler::ProfilerPtr
std::shared_ptr< Profiler > ProfilerPtr
Definition: ManagedIceObject.h:74
armarx::ProcessWatcher::reportCpuUsage
void reportCpuUsage()
Definition: ProcessWatcher.cpp:376
armarx::ProcessWatcher::start
void start()
Definition: ProcessWatcher.cpp:65
armarx::PeriodicTask
Definition: ArmarXManager.h:70
Logging.h
armarx::ThreadUsage
Definition: ProcessWatcher.h:41
armarx::MemoryUsage::totalFreeSpace
int totalFreeSpace
Definition: ProcessWatcher.h:67
armarx
This file offers overloads of toIce() and fromIce() functions for STL container types.
Definition: ArmarXTimeserver.cpp:27
armarx::CpuUsage::lastUpdate
IceUtil::Time lastUpdate
Definition: ProcessWatcher.h:59
armarx::ThreadUsage::operator==
bool operator==(const ThreadUsage &rhs) const
Definition: ProcessWatcher.cpp:350
armarx::split
std::vector< std::string > split(const std::string &source, const std::string &splitBy, bool trimElements=false, bool removeEmptyElements=false)
Definition: StringHelpers.cpp:38