support.cpp
1 /***************************************************************************
2  Copyright (C) 2002-2015 Kentaro Kitagawa
3  kitagawa@phys.s.u-tokyo.ac.jp
4 
5  This program is free software; you can redistribute it and/or
6  modify it under the terms of the GNU Library General Public
7  License as published by the Free Software Foundation; either
8  version 2 of the License, or (at your option) any later version.
9 
10  You should have received a copy of the GNU Library General
11  Public License and a list of authors along with this program;
12  see the files COPYING and AUTHORS.
13 ***************************************************************************/
14 #include <errno.h>
15 #include <fcntl.h>
16 #include <string.h>
17 #ifndef _MSC_VER
18  #include <unistd.h>
19  #include <cpuid.h>
20 #endif
21 
22 #include "support.h"
23 
24 bool g_bLogDbgPrint;
25 bool g_bUseOverpaint;
26 bool g_bMLockAlways;
27 bool g_bUseMLock;
28 
29 bool isMemLockAvailable() noexcept {return g_bUseMLock;}
30 
31 #include <iostream>
32 #include <fstream>
33 
34 #include "xthread.h"
35 
36 #if defined __WIN32__ || defined WINDOWS || defined _WIN32
37  #define KAME_LOG_FILENAME "kame.log"
38 #else
39  #define KAME_LOG_FILENAME "/tmp/kame.log"
40 #endif
41 
42 static std::ofstream g_debugofs(KAME_LOG_FILENAME, std::ios::out);
43 static XMutex g_debug_mutex;
44 
45 #include "xtime.h"
46 #include "measure.h"
47 #include "threadlocal.h"
48 
49 #if defined __linux__ || defined __APPLE__
50 #undef TRAP_FPE
51 #if defined TRAP_FPE && defined __linux__
52 #include <fpu_control.h>
53 static void __attribute__ ((constructor)) trapfpe (void)
54 {
55  fpu_control_t cw =
56  _FPU_DEFAULT & ~(_FPU_MASK_IM | _FPU_MASK_ZM | _FPU_MASK_OM);
57  _FPU_SETCW(cw);
58 }
59 #endif
60 #endif // __linux__
61 
62 XKameError::XKameError() : std::runtime_error(""), m_msg(""), m_file(0), m_line(0), m_errno(0) {
63 
64 }
65 XKameError::XKameError(const XString &s, const char *file, int line)
66  : std::runtime_error(s.c_str()), m_msg(s), m_file(file), m_line(line), m_errno(errno) {
67  errno = 0;
68 }
69 
70 void
71 XKameError::print(const XString &header) {
72  print(header + m_msg, m_file, m_line, m_errno);
73 }
74 void
75 XKameError::print() {
76  print("");
77 }
78 void
79 XKameError::print(const XString &msg, const char *file, int line, int errno_) {
80  if( !file) return;
81  if(errno_) {
82  errno = 0;
83  char buf[256];
84  #ifdef __linux__
85  char *s = strerror_r(errno_, buf, sizeof(buf));
86  gErrPrint_redirected(msg + " " + s, file, line);
87  #else
88  #if defined __WIN32__ || defined WINDOWS || defined _WIN32
89  strerror_s(buf, sizeof(buf), errno_);
90  #else
91  strerror_r(errno_, buf, sizeof(buf));
92  #endif
93  if( !errno_)
94  gErrPrint_redirected(msg + " " + buf, file, line);
95  else
96  gErrPrint_redirected(msg + " (strerror failed)", file, line);
97  #endif
98  errno = 0;
99  }
100  else {
101  gErrPrint_redirected(msg, file, line);
102  }
103 }
104 
105 const XString &
106 XKameError::msg() const {
107  return m_msg;
108 }
109 
110 const char* XKameError::what() const noexcept {
111  return m_msg.c_str();
112 }
113 
114 double roundlog10(double val) {
115  int i = lrint(log10(val));
116  return pow(10.0, (double)i);
117 }
118 double setprec(double val, double prec) noexcept {
119  double x;
120 
121  if(prec <= 1e-100) return val;
122  x = roundlog10(prec/2);
123  double f = rint(val / x);
124  double z = (fabs(f) < (double)0x8fffffff) ? ((int)f) * x : f * x;
125  return z;
126 }
127 
128 
129 //---------------------------------------------------------------------------
130 #include "xtime.h"
131 
132 
133 void
134 dbgPrint_redirected(const XString &str, const char *file, int line, bool force_dump) {
135  if( !force_dump && !g_bLogDbgPrint) return;
136  XScopedLock<XMutex> lock(g_debug_mutex);
137  g_debugofs
138  << (QString("0x%1:%2:%3:%4 %5")
139  .arg((uintptr_t)threadID(), 0, 16)
140  .arg(XTime::now().getTimeStr())
141  .arg(file)
142  .arg(line)
143  .arg(str)).toUtf8().data()
144  << std::endl;
145  if(force_dump) {
146  shared_ptr<XStatusPrinter> statusprinter = g_statusPrinter;
147  if(statusprinter) statusprinter->printMessage(str, true, file, line);
148  }
149 }
150 void
151 gErrPrint_redirected(const XString &str, const char *file, int line) {
152  {
153  XScopedLock<XMutex> lock(g_debug_mutex);
154  fprintf(stderr, "err:%s:%d %s\n", file, line, (const char*)QString(str).toLocal8Bit().data());
155  g_debugofs
156  << (const char*)(QString("Err:0x%1:%2:%3:%4 %5")
157  .arg((uintptr_t)threadID(), 0, 16)
158  .arg(XTime::now().getTimeStr())
159  .arg(file)
160  .arg(line)
161  .arg(str)).toUtf8().data()
162  << std::endl;
163 #if !defined __WIN32__ && !defined WINDOWS && !defined _WIN32
164  sync(); //ensures disk writing.
165 #endif
166  }
167  shared_ptr<XStatusPrinter> statusprinter = g_statusPrinter;
168  if(statusprinter) statusprinter->printError(str, true, file, line);
169 }
170 void
171 gWarnPrint_redirected(const XString &str, const char *file, int line) {
172  {
173  XScopedLock<XMutex> lock(g_debug_mutex);
174  fprintf(stderr, "warn:%s:%d %s\n", file, line, (const char*)QString(str).toLocal8Bit().data());
175  g_debugofs
176  << (const char*)(QString("Warn:0x%1:%2:%3:%4 %5")
177  .arg((uintptr_t)threadID(), 0, 16)
178  .arg(XTime::now().getTimeStr())
179  .arg(file)
180  .arg(line)
181  .arg(str)).toUtf8().data()
182  << std::endl;
183  }
184  shared_ptr<XStatusPrinter> statusprinter = g_statusPrinter;
185  if(statusprinter) statusprinter->printWarning(str, false, file, line);
186 }
187 
188 #define SNPRINT_BUF_SIZE 128
189 #include <stdarg.h>
190 #include <vector>
191 
192 static XString
193 v_formatString(const char *fmt, va_list ap) {
194  int buf_size = SNPRINT_BUF_SIZE;
195  std::vector<char> buf;
196  for(;;) {
197  buf.resize(buf_size);
198  int ret;
199 
200  ret = vsnprintf(&buf[0], buf_size, fmt, ap);
201 
202  if(ret < 0) throw XKameError(i18n_noncontext("Mal-format conversion."), __FILE__, __LINE__);
203  if(ret < buf_size) break;
204 
205  buf_size *= 2;
206  }
207  return XString((char*)&buf[0]);
208 }
209 
210 XString
211 formatString_tr(const char *fmt, ...) {
212  va_list ap;
213  va_start(ap, fmt);
214  XString str = v_formatString(i18n_noncontext(fmt).toUtf8().data(), ap);
215  va_end(ap);
216  return str;
217 }
218 
219 XString
220 formatString(const char *fmt, ...) {
221  va_list ap;
222  va_start(ap, fmt);
223  XString str = v_formatString(fmt, ap);
224  va_end(ap);
225  return str;
226 }
227 
228 XString formatDouble(const char *fmt, double var) {
229  char cbuf[SNPRINT_BUF_SIZE];
230  if(strlen(fmt) == 0) {
231  snprintf(cbuf, sizeof(cbuf), "%.12g", var);
232  return XString(cbuf);
233  }
234 
235  if(!strncmp(fmt, "TIME:", 5)) {
236 #if !defined __WIN32__ && !defined WINDOWS && !defined _WIN32
237  if(isnan(var))
238  return "nan";
239 #endif
240  XTime time;
241  time += var;
242  if(fmt[5])
243  return time.getTimeFmtStr(fmt + 5, false);
244  else
245  return time.getTimeStr(false);
246  }
247  snprintf(cbuf, sizeof(cbuf), fmt, var);
248  return XString(cbuf);
249 }
250 void formatDoubleValidator(XString &fmt) {
251  if(fmt.empty()) return;
252 
253  XString buf(fmt);
254 
255  if( !strncmp(buf.c_str(), "TIME:", 5)) return;
256 
257  int arg_cnt = 0;
258  for(int pos = 0;;) {
259  pos = buf.find('%', pos);
260  if(pos == std::string::npos) break;
261  pos++;
262  if(buf[pos] == '%') {
263  continue;
264  }
265  arg_cnt++;
266  if(arg_cnt > 1) {
267  throw XKameError(i18n_noncontext("Illegal Format, too many %s."), __FILE__, __LINE__);
268  }
269  char conv;
270  if((sscanf(buf.c_str() + pos, "%*[+-'0# ]%*f%c", &conv) != 1) &&
271  (sscanf(buf.c_str() + pos, "%*[+-'0# ]%c", &conv) != 1) &&
272  (sscanf(buf.c_str() + pos, "%*f%c", &conv) != 1) &&
273  (sscanf(buf.c_str() + pos, "%c", &conv) != 1)) {
274  throw XKameError(i18n_noncontext("Illegal Format."), __FILE__, __LINE__);
275  }
276  if(std::string("eEgGf").find(conv) == std::string::npos)
277  throw XKameError(i18n_noncontext("Illegal Format, no float conversion."), __FILE__, __LINE__);
278  }
279  if(arg_cnt == 0)
280  throw XKameError(i18n_noncontext("Illegal Format, no %."), __FILE__, __LINE__);
281 }
282 
283 XString dumpCString(const char *cstr) {
284  XString buf;
285  for(; *cstr; cstr++) {
286  if(isprint(*cstr))
287  buf.append(1, *cstr);
288  else {
289  char s[5];
290  snprintf(s, 5, "\\x%02x", (unsigned int)(int)*cstr);
291  buf.append(s);
292  }
293  }
294  return buf;
295 }
296 
297 #if defined __WIN32__ || defined WINDOWS || defined _WIN32
298  #include <windows.h>
299  int mlock(const void *addr, size_t len) {
300  return (VirtualLock((LPVOID)addr, len) != 0) ? 0 : -1;
301  }
302 #endif
303 
304 #if defined __i386__ || defined __i486__ || defined __i586__ || defined __i686__ || defined __x86_64__
305 X86CPUSpec::X86CPUSpec() {
306  uint32_t stepinfo, brand, features_ext, features;
307 #ifdef _MSC_VER
308  uint32_t cpuinfo[4];
309  __cpuid(reinterpret_cast<int*>(cpuinfo), 0x1);
310  stepinfo = cpuinfo[0];
311  brand = cpuinfo[1];
312  features_ext = cpuinfo[2];
313  features = cpuinfo[3];
314 #else
315  __cpuid(0x1, stepinfo, brand , features_ext, features);
316 // #if defined __LP64__ || defined __LLP64__
317 // asm volatile("push %%rbx; cpuid; pop %%rbx"
318 // #else
319 // asm volatile("push %%ebx; cpuid; pop %%ebx"
320 // #endif
321 // : "=a" (stepinfo), "=c" (features_ext), "=d" (features) : "a" (0x1));
322 #endif
323  verSSE = (features & (1uL << 25)) ? 1 : 0;
324  if(verSSE && (features & (1uL << 26)))
325  verSSE = 2;
326  if((verSSE == 2) && (features_ext & (1uL << 0)))
327  verSSE = 3;
328 #ifdef __APPLE__
329  hasMonitor = false;
330 #else
331  hasMonitor = (verSSE == 3) && (features_ext & (1uL << 3));
332 #endif
333  monitorSizeSmallest = 0L;
334  monitorSizeLargest = 0L;
335  if(hasMonitor) {
336  uint32_t monsize_s, monsize_l;
337  uint32_t cpuinfo[4];
338 #ifdef _MSC_VER
339  __cpuid(reinterpret_cast<int*>(cpuinfo), 0x5);
340  monsize_s = cpuinfo[0];
341  monsize_l = cpuinfo[2];
342 #else
343  __cpuid(0x5, monsize_s, cpuinfo[1] , monsize_l, cpuinfo[2]);
344  //#if defined __LP64__ || defined __LLP64__
345  // asm volatile("push %%rbx; cpuid; mov %%ebx, %%ecx; pop %%rbx"
346  //#else
347  // asm volatile("push %%ebx; cpuid; mov %%ebx, %%ecx; pop %%ebx"
348  //#endif
349  // : "=a" (monsize_s), "=c" (monsize_l) : "a" (0x5) : "%edx");
350 #endif
351  monitorSizeSmallest = monsize_s;
352  monitorSizeLargest = monsize_l;
353  }
354  fprintf(stderr, "Target: "
355 #if defined __LP64__
356  "x86-64, LP64"
357 #else
358  #if defined __LLP64__
359  "x86-64, LLP64"
360  #else
361  "x86-32"
362  #endif
363  #if defined __SSE2__
364  ", SSE2"
365  #endif
366 #endif
367  "; Detected: SSE%u\n", verSSE);
368 }
369 const X86CPUSpec cg_cpuSpec;
370 #endif
371 
372 
373 
374 

Generated for KAME4 by  doxygen 1.8.3