2 #include "ace/OS_NS_string.h"
3 #include "ace/OS_NS_errno.h"
4 #include "ace/Time_Value.h"
5 #include "ace/OS_NS_unistd.h"
6 #include "ace/OS_NS_sys_time.h"
9 ACE_BEGIN_VERSIONED_NAMESPACE_DECL
12 ACE_OS::asctime (const struct tm *t)
14 ACE_OS_TRACE ("ACE_OS::asctime");
15 #if defined (ACE_LACKS_ASCTIME)
17 ACE_NOTSUP_RETURN (0);
19 return std::asctime (t);
20 #endif /* ACE_LACKS_ASCTIME */
24 ACE_OS::asctime_r (const struct tm *t, char *buf, int buflen)
26 ACE_OS_TRACE ("ACE_OS::asctime_r");
27 #if defined (ACE_HAS_REENTRANT_FUNCTIONS)
28 # if defined (ACE_HAS_2_PARAM_ASCTIME_R_AND_CTIME_R)
30 ace_asctime_r_helper (t, buf);
31 ACE_OS::strsncpy (buf, result, buflen);
34 # if defined (ACE_HAS_SIZET_PTR_ASCTIME_R_AND_CTIME_R)
35 return ::asctime_r (t, buf, reinterpret_cast<size_t*>(&buflen));
37 return ::asctime_r (t, buf, buflen);
38 # endif /* ACE_HAS_SIZET_PTR_ASCTIME_R_AND_CTIME_R */
39 # endif /* ACE_HAS_2_PARAM_ASCTIME_R_AND_CTIME_R */
40 #elif defined (ACE_HAS_TR24731_2005_CRT)
42 ACE_SECURECRTCALL (asctime_s (buf, static_cast<size_t> (buflen), t), \
45 #elif defined (ACE_LACKS_ASCTIME)
48 ACE_UNUSED_ARG (buflen);
49 ACE_NOTSUP_RETURN (0);
52 ACE_OSCALL (std::asctime (t), char *, result);
53 ACE_OS::strsncpy (buf, result, buflen);
55 #endif /* ACE_HAS_REENTRANT_FUNCTIONS */
59 ACE_OS::clock_gettime (clockid_t clockid, struct timespec *ts)
61 ACE_OS_TRACE ("ACE_OS::clock_gettime");
62 #if defined (ACE_HAS_CLOCK_GETTIME)
63 return ::clock_gettime (clockid, ts);
65 ACE_UNUSED_ARG (clockid);
67 ACE_NOTSUP_RETURN (-1);
68 #endif /* ACE_HAS_CLOCK_GETTIME */
72 ACE_OS::clock_settime (clockid_t clockid, const struct timespec *ts)
74 #if defined (ACE_HAS_CLOCK_SETTIME)
75 # if defined (ACE_HAS_NONCONST_CLOCK_SETTIME)
76 return ::clock_settime (clockid, const_cast<struct timespec *>(ts));
78 return ::clock_settime (clockid, ts);
79 # endif /* ACE_HAS_NONCONST_CLOCK_SETTIME */
81 ACE_UNUSED_ARG (clockid);
83 ACE_NOTSUP_RETURN (-1);
84 #endif /* ACE_HAS_CLOCK_SETTIME */
87 // Magic number declaration and definition for ctime and ctime_r ()
88 static constexpr int ctime_buf_size = 26;
90 ACE_INLINE ACE_TCHAR *
91 ACE_OS::ctime (const time_t *t)
93 ACE_OS_TRACE ("ACE_OS::ctime");
94 #if defined (ACE_LACKS_CTIME)
96 ACE_NOTSUP_RETURN (0);
97 #elif defined (ACE_WIN32) && defined (ACE_USES_WCHAR)
100 # if defined (ACE_USES_WCHAR) /* Not Win32, else it would do the above */
102 ACE_OSCALL (::ctime (t), char *, narrow_time);
103 if (narrow_time == 0)
105 // ACE_Ascii_To_Wide::convert allocates (via new []) a wchar_t[]. If
106 // we've done this before, free the previous one. Yes, this leaves a
107 // small memory leak (26 characters) but there's no way around this
108 // that I know of. (Steve Huston, 12-Feb-2003).
109 static wchar_t *wide_time = nullptr;
110 if (wide_time != nullptr)
112 wide_time = ACE_Ascii_To_Wide::convert (narrow_time);
116 # endif /* ACE_USES_WCHAR */
117 # endif /* ACE_LACKS_CTIME */
120 ACE_INLINE ACE_TCHAR *
121 ACE_OS::ctime_r (const time_t *t, ACE_TCHAR *buf, int buflen)
123 ACE_OS_TRACE ("ACE_OS::ctime_r");
125 #if defined (ACE_HAS_REENTRANT_FUNCTIONS)
127 char *bufp = nullptr;
128 # if defined (ACE_USES_WCHAR)
129 char narrow_buf[ctime_buf_size];
133 # endif /* ACE_USES_WCHAR */
135 if (buflen < ctime_buf_size)
140 # if defined (ACE_HAS_2_PARAM_ASCTIME_R_AND_CTIME_R)
141 ACE_OSCALL (::ctime_r (t, bufp), char *, bufp);
142 # else /* ACE_HAS_2_PARAM_ASCTIME_R_AND_CTIME_R */
144 # if defined (ACE_HAS_SIZET_PTR_ASCTIME_R_AND_CTIME_R)
145 bufp = ::ctime_r (t, bufp, reinterpret_cast<size_t*>(&buflen));
146 # else /* ACE_CTIME_R_RETURNS_INT */
147 bufp = ::ctime_r (t, bufp, buflen);
148 # endif /* ACE_CTIME_R_RETURNS_INT */
150 # endif /* ACE_HAS_2_PARAM_ASCTIME_R_AND_CTIME_R */
155 # if defined (ACE_USES_WCHAR)
156 ACE_Ascii_To_Wide wide_buf (bufp);
157 ACE_OS::strcpy (buf, wide_buf.wchar_rep ());
161 # endif /* ACE_USES_WCHAR */
163 #elif defined (ACE_HAS_TR24731_2005_CRT)
164 if (buflen < ctime_buf_size)
169 ACE_TCHAR *result = buf;
170 # if defined (ACE_USES_WCHAR)
171 ACE_SECURECRTCALL (_wctime_s (buf, buflen, t), wchar_t *, 0, result);
173 ACE_SECURECRTCALL (ctime_s (buf, buflen, t), char *, 0, result);
177 #else /* ACE_HAS_REENTRANT_FUNCTIONS */
178 if (buflen < ctime_buf_size)
184 ACE_TCHAR *result = nullptr;
185 # if defined (ACE_USES_WCHAR)
186 ACE_OSCALL (::_wctime (t), wchar_t *, result);
187 # else /* ACE_USES_WCHAR */
188 ACE_OSCALL (::ctime (t), char *, result);
189 # endif /* ACE_USES_WCHAR */
190 if (result != nullptr)
191 ACE_OS::strsncpy (buf, result, buflen);
193 #endif /* ACE_HAS_REENTRANT_FUNCTIONS */
196 #if defined (ACE_USES_ULONG_FOR_STAT_TIME)
197 ACE_INLINE ACE_TCHAR *
198 ACE_OS::ctime (const unsigned long *t)
200 return ACE_OS::ctime (reinterpret_cast<const time_t *>(t));
203 ACE_INLINE ACE_TCHAR *
204 ACE_OS::ctime_r (const unsigned long *clock, ACE_TCHAR *buf, int buflen)
206 return ACE_OS::ctime_r (reinterpret_cast<const time_t *>(clock), buf, buflen);
209 ACE_INLINE struct tm *
210 ACE_OS::gmtime (const unsigned long *clock)
212 return ACE_OS::gmtime (reinterpret_cast<const time_t *>(clock));
215 ACE_INLINE struct tm *
216 ACE_OS::gmtime_r (const unsigned long *clock,
219 return ACE_OS::gmtime_r (reinterpret_cast<const time_t *>(clock), res);
222 ACE_INLINE struct tm *
223 ACE_OS::localtime (const unsigned long *clock)
225 return ACE_OS::localtime (reinterpret_cast<const time_t *>(clock));
228 ACE_INLINE struct tm *
229 ACE_OS::localtime_r (const unsigned long *clock,
232 return ACE_OS::localtime_r (reinterpret_cast<const time_t *>(clock), res);
239 ACE_OS::difftime (time_t t1, time_t t0)
241 return ::ace_difftime (t1, t0);
244 ACE_INLINE ACE_hrtime_t
245 ACE_OS::gethrtime (const ACE_HRTimer_Op op)
247 ACE_OS_TRACE ("ACE_OS::gethrtime");
248 #if defined (ACE_HAS_HI_RES_TIMER)
250 return ::gethrtime ();
251 #elif defined (ACE_WIN32)
255 ::QueryPerformanceCounter (&freq);
257 return freq.QuadPart;
258 #elif defined (ghs) && defined (ACE_HAS_PENTIUM)
260 // Use .obj/gethrtime.o, which was compiled with g++.
261 return ACE_GETHRTIME_NAME ();
262 #elif (defined (__GNUG__) || defined (__INTEL_COMPILER)) && \
263 !defined (ACE_VXWORKS) && defined (ACE_HAS_PENTIUM) && \
264 !defined (ACE_LACKS_PENTIUM_RDTSC)
268 # if defined (__amd64__) || defined (__x86_64__)
269 // Read the high res tick counter into 32 bit int variables "eax" and
270 // "edx", and then combine them into 64 bit int "now"
272 asm volatile ("rdtsc" : "=a" (eax), "=d" (edx) : : "memory");
273 now = (((ACE_UINT64) eax) | (((ACE_UINT64) edx) << 32));
275 // Read the high-res tick counter directly into memory variable "now".
276 // The A constraint signifies a 64-bit int.
277 asm volatile ("rdtsc" : "=A" (now) : : "memory");
281 #elif defined (ACE_HAS_POWERPC_TIMER) && (defined (ghs) || defined (__GNUG__))
282 // PowerPC w/ GreenHills or g++.
289 ACE_OS::readPPCTimeBase (most, least);
294 asm volatile ("mftbu %0\n"
297 : "=r" (most), "=r" (least), "=r" (scratch));
298 } while (most != scratch);
301 return 0x100000000llu * most + least;
303 #elif defined (ACE_HAS_CLOCK_GETTIME)
304 // e.g., VxWorks (besides POWERPC && GreenHills) . . .
308 ACE_OS::clock_gettime (
309 # if defined (ACE_HAS_CLOCK_GETTIME_MONOTONIC)
313 # endif /* !ACE_HAS_CLOCK_GETTIME_MONOTONIC */
316 // Carefully create the return value to avoid arithmetic overflow
317 return static_cast<ACE_hrtime_t> (ts.tv_sec) *
318 ACE_U_ONE_SECOND_IN_NSECS + static_cast<ACE_hrtime_t> (ts.tv_nsec);
321 ACE_Time_Value const now = ACE_OS::gettimeofday ();
323 // Carefully create the return value to avoid arithmetic overflow
324 return (static_cast<ACE_hrtime_t> (now.sec ()) * (ACE_UINT32) 1000000 +
325 static_cast<ACE_hrtime_t> (now.usec ())) * (ACE_UINT32) 1000;
326 #endif /* ACE_HAS_HI_RES_TIMER */
329 ACE_INLINE struct tm *
330 ACE_OS::gmtime (const time_t *t)
332 ACE_OS_TRACE ("ACE_OS::gmtime");
333 #if defined (ACE_LACKS_GMTIME)
335 ACE_NOTSUP_RETURN (0);
338 #endif /* ACE_LACKS_GMTIME */
341 ACE_INLINE struct tm *
342 ACE_OS::gmtime_r (const time_t *t, struct tm *res)
344 ACE_OS_TRACE ("ACE_OS::gmtime_r");
345 #if defined (ACE_HAS_REENTRANT_FUNCTIONS)
346 return ace_gmtime_r_helper (t, res);
347 #elif defined (ACE_HAS_TR24731_2005_CRT)
348 struct tm *tm_p = res;
349 ACE_SECURECRTCALL (gmtime_s (res, t), struct tm *, 0, tm_p);
351 #elif defined (ACE_LACKS_GMTIME)
353 ACE_UNUSED_ARG (res);
354 ACE_NOTSUP_RETURN (0);
357 ACE_OSCALL (::gmtime (t), struct tm *, result) ;
361 #endif /* ACE_HAS_REENTRANT_FUNCTIONS */
364 ACE_INLINE struct tm *
365 ACE_OS::localtime (const time_t *t)
367 ACE_OS_TRACE ("ACE_OS::localtime");
368 #if defined (ACE_LACKS_LOCALTIME)
370 ACE_NOTSUP_RETURN (0);
372 return ::localtime (t);
373 #endif /* ACE_LACKS_LOCALTIME */
377 ACE_OS::nanosleep (const struct timespec *requested,
378 struct timespec *remaining)
380 ACE_OS_TRACE ("ACE_OS::nanosleep");
381 #if defined (ACE_HAS_CLOCK_GETTIME)
382 // ::nanosleep () is POSIX 1003.1b. So is ::clock_gettime (). So,
383 // if ACE_HAS_CLOCK_GETTIME is defined, then ::nanosleep () should
384 // be available on the platform
385 return ::nanosleep ((ACE_TIMESPEC_PTR) requested, remaining);
387 ACE_UNUSED_ARG (remaining);
389 // Convert into seconds and microseconds.
390 ACE_Time_Value tv (requested->tv_sec,
391 requested->tv_nsec / 1000);
392 return ACE_OS::sleep (tv);
393 #endif /* ACE_HAS_CLOCK_GETTIME */
397 ACE_OS::strftime (char *s, size_t maxsize, const char *format,
398 const struct tm *timeptr)
400 #if defined (ACE_LACKS_STRFTIME)
402 ACE_UNUSED_ARG (maxsize);
403 ACE_UNUSED_ARG (format);
404 ACE_UNUSED_ARG (timeptr);
405 ACE_NOTSUP_RETURN (0);
407 return std::strftime (s, maxsize, format, timeptr);
408 #endif /* ACE_LACKS_STRFTIME */
412 ACE_OS::strptime (const char *buf, const char *format, struct tm *tm)
414 ACE_OS::memset (tm, 0, sizeof (struct tm));
415 #if defined (ACE_LACKS_STRPTIME)
416 return ACE_OS::strptime_emulation (buf, format, tm);
418 return ::strptime (buf, format, tm);
419 #endif /* ACE_LACKS_STRPTIME */
423 ACE_OS::time (time_t *tloc)
425 ACE_OS_TRACE ("ACE_OS::time");
426 #if defined (ACE_LACKS_TIME)
427 time_t const retv = ACE_OS::gettimeofday ().sec ();
432 return ::time (tloc);
433 #endif /* ACE_LACKS_TIME */
436 // Linux won't compile unless we explicitly use a namespace here.
437 #if defined (__GNUG__)
442 return ::ace_timezone ();
444 } /* namespace ACE_OS */
449 return ::ace_timezone ();
451 #endif /* ACE_LINUX */
456 #if defined (ACE_LACKS_TZSET)
458 #elif defined (ACE_WIN32)
459 ::_tzset (); // For Win32.
461 ::tzset (); // For UNIX platforms.
462 #endif /* ACE_LACKS_TZSET */
465 ACE_END_VERSIONED_NAMESPACE_DECL