521 lines
11 KiB
C++
521 lines
11 KiB
C++
// Module: Log4CPLUS
|
|
// File: timehelper.cxx
|
|
// Created: 4/2003
|
|
// Author: Tad E. Smith
|
|
//
|
|
//
|
|
// Copyright 2003-2009 Tad E. Smith
|
|
//
|
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
|
// you may not use this file except in compliance with the License.
|
|
// You may obtain a copy of the License at
|
|
//
|
|
// http://www.apache.org/licenses/LICENSE-2.0
|
|
//
|
|
// Unless required by applicable law or agreed to in writing, software
|
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
// See the License for the specific language governing permissions and
|
|
// limitations under the License.
|
|
|
|
#include <log4cplus/helpers/timehelper.h>
|
|
#include <log4cplus/helpers/loglog.h>
|
|
#include <log4cplus/streams.h>
|
|
#include <log4cplus/helpers/stringhelper.h>
|
|
|
|
#include <ctime>
|
|
#include <algorithm>
|
|
#include <stdexcept>
|
|
#include <vector>
|
|
#include <iomanip>
|
|
#include <cassert>
|
|
#if ! defined (_WIN32_WCE)
|
|
#include <cerrno>
|
|
#endif
|
|
#if defined (UNICODE)
|
|
#include <cwchar>
|
|
#endif
|
|
|
|
#if defined (LOG4CPLUS_HAVE_SYS_TYPES_H)
|
|
#include <sys/types.h>
|
|
#endif
|
|
|
|
#if defined(LOG4CPLUS_HAVE_FTIME)
|
|
#include <sys/timeb.h>
|
|
#endif
|
|
|
|
#if defined(LOG4CPLUS_HAVE_GETTIMEOFDAY)
|
|
#include <sys/time.h>
|
|
#endif
|
|
|
|
#if defined(LOG4CPLUS_HAVE_GMTIME_R) && !defined(LOG4CPLUS_SINGLE_THREADED)
|
|
#define LOG4CPLUS_NEED_GMTIME_R
|
|
#endif
|
|
|
|
#if defined(LOG4CPLUS_HAVE_LOCALTIME_R) && !defined(LOG4CPLUS_SINGLE_THREADED)
|
|
#define LOG4CPLUS_NEED_LOCALTIME_R
|
|
#endif
|
|
|
|
|
|
namespace log4cplus { namespace helpers {
|
|
|
|
const int ONE_SEC_IN_USEC = 1000000;
|
|
|
|
|
|
#if defined (_WIN32_WCE)
|
|
using ::mktime;
|
|
using ::gmtime;
|
|
using ::localtime;
|
|
#if defined (UNICODE)
|
|
using ::wcsftime;
|
|
#else
|
|
using ::strftime;
|
|
#endif
|
|
|
|
#else
|
|
using std::mktime;
|
|
using std::gmtime;
|
|
using std::localtime;
|
|
#if defined (UNICODE)
|
|
using std::wcsftime;
|
|
#else
|
|
using std::strftime;
|
|
#endif
|
|
|
|
#endif
|
|
|
|
|
|
|
|
//////////////////////////////////////////////////////////////////////////////
|
|
// Time ctors
|
|
//////////////////////////////////////////////////////////////////////////////
|
|
|
|
Time::Time()
|
|
: tv_sec(0),
|
|
tv_usec(0)
|
|
{
|
|
}
|
|
|
|
|
|
Time::Time(time_t tv_sec_, long tv_usec_)
|
|
: tv_sec(tv_sec_),
|
|
tv_usec(tv_usec_)
|
|
{
|
|
assert (tv_usec < ONE_SEC_IN_USEC);
|
|
}
|
|
|
|
|
|
Time::Time(time_t time)
|
|
: tv_sec(time),
|
|
tv_usec(0)
|
|
{
|
|
}
|
|
|
|
|
|
Time
|
|
Time::gettimeofday()
|
|
{
|
|
#if defined (LOG4CPLUS_HAVE_CLOCK_GETTIME)
|
|
struct timespec ts;
|
|
int res = clock_gettime (CLOCK_REALTIME, &ts);
|
|
assert (res == 0);
|
|
if (res != 0)
|
|
throw std::runtime_error ("clock_gettime() has failed");
|
|
|
|
return Time (ts.tv_sec, ts.tv_nsec / 1000);
|
|
#elif defined(LOG4CPLUS_HAVE_GETTIMEOFDAY)
|
|
timeval tp;
|
|
::gettimeofday(&tp, 0);
|
|
|
|
return Time(tp.tv_sec, tp.tv_usec);
|
|
#elif defined(LOG4CPLUS_HAVE_FTIME)
|
|
struct timeb tp;
|
|
::ftime(&tp);
|
|
|
|
return Time(tp.time, tp.millitm * 1000);
|
|
#else
|
|
#warning "Time::gettimeofday()- low resolution timer: gettimeofday and ftime unavailable"
|
|
return Time(::time(0), 0);
|
|
#endif
|
|
}
|
|
|
|
|
|
//////////////////////////////////////////////////////////////////////////////
|
|
// Time methods
|
|
//////////////////////////////////////////////////////////////////////////////
|
|
|
|
time_t
|
|
Time::setTime(tm* t)
|
|
{
|
|
time_t time = helpers::mktime(t);
|
|
if (time != -1)
|
|
tv_sec = time;
|
|
|
|
return time;
|
|
}
|
|
|
|
|
|
time_t
|
|
Time::getTime() const
|
|
{
|
|
return tv_sec;
|
|
}
|
|
|
|
|
|
void
|
|
Time::gmtime(tm* t) const
|
|
{
|
|
time_t clock = tv_sec;
|
|
#ifdef LOG4CPLUS_NEED_GMTIME_R
|
|
::gmtime_r(&clock, t);
|
|
#else
|
|
tm* tmp = helpers::gmtime(&clock);
|
|
*t = *tmp;
|
|
#endif
|
|
}
|
|
|
|
|
|
void
|
|
Time::localtime(tm* t) const
|
|
{
|
|
time_t clock = tv_sec;
|
|
#ifdef LOG4CPLUS_NEED_LOCALTIME_R
|
|
::localtime_r(&clock, t);
|
|
#else
|
|
tm* tmp = helpers::localtime(&clock);
|
|
*t = *tmp;
|
|
#endif
|
|
}
|
|
|
|
|
|
namespace
|
|
{
|
|
|
|
static log4cplus::tstring const padding_zeros[4] =
|
|
{
|
|
log4cplus::tstring (LOG4CPLUS_TEXT("000")),
|
|
log4cplus::tstring (LOG4CPLUS_TEXT("00")),
|
|
log4cplus::tstring (LOG4CPLUS_TEXT("0")),
|
|
log4cplus::tstring (LOG4CPLUS_TEXT(""))
|
|
};
|
|
|
|
static log4cplus::tstring const uc_q_padding_zeros[4] =
|
|
{
|
|
log4cplus::tstring (LOG4CPLUS_TEXT(".000")),
|
|
log4cplus::tstring (LOG4CPLUS_TEXT(".00")),
|
|
log4cplus::tstring (LOG4CPLUS_TEXT(".0")),
|
|
log4cplus::tstring (LOG4CPLUS_TEXT("."))
|
|
};
|
|
|
|
}
|
|
|
|
|
|
void
|
|
Time::build_q_value (log4cplus::tstring & q_str) const
|
|
{
|
|
q_str = convertIntegerToString(tv_usec / 1000);
|
|
size_t const len = q_str.length();
|
|
if (len <= 2)
|
|
q_str.insert (0, padding_zeros[q_str.length()]);
|
|
}
|
|
|
|
|
|
void
|
|
Time::build_uc_q_value (log4cplus::tstring & uc_q_str) const
|
|
{
|
|
build_q_value (uc_q_str);
|
|
|
|
log4cplus::tstring usecs (convertIntegerToString(tv_usec % 1000));
|
|
size_t usecs_len = usecs.length();
|
|
usecs.insert (0, usecs_len <= 3
|
|
? uc_q_padding_zeros[usecs_len] : uc_q_padding_zeros[3]);
|
|
uc_q_str.append (usecs);
|
|
}
|
|
|
|
|
|
log4cplus::tstring
|
|
Time::getFormattedTime(const log4cplus::tstring& fmt_orig, bool use_gmtime) const
|
|
{
|
|
if (fmt_orig.empty () || fmt_orig[0] == 0)
|
|
return log4cplus::tstring ();
|
|
|
|
tm time;
|
|
|
|
if(use_gmtime)
|
|
gmtime(&time);
|
|
else
|
|
localtime(&time);
|
|
|
|
enum State
|
|
{
|
|
TEXT,
|
|
PERCENT_SIGN
|
|
};
|
|
|
|
log4cplus::tstring fmt (fmt_orig);
|
|
log4cplus::tstring ret;
|
|
ret.reserve (static_cast<size_t>(fmt.size () * 1.35));
|
|
State state = TEXT;
|
|
|
|
log4cplus::tstring q_str;
|
|
bool q_str_valid = false;
|
|
|
|
log4cplus::tstring uc_q_str;
|
|
bool uc_q_str_valid = false;
|
|
|
|
log4cplus::tstring s_str;
|
|
bool s_str_valid = false;
|
|
|
|
// Walk the format string and process all occurences of %q and %Q.
|
|
|
|
for (log4cplus::tstring::const_iterator fmt_it = fmt.begin ();
|
|
fmt_it != fmt.end (); ++fmt_it)
|
|
{
|
|
switch (state)
|
|
{
|
|
case TEXT:
|
|
{
|
|
if (*fmt_it == LOG4CPLUS_TEXT ('%'))
|
|
state = PERCENT_SIGN;
|
|
else
|
|
ret.push_back (*fmt_it);
|
|
}
|
|
break;
|
|
|
|
case PERCENT_SIGN:
|
|
{
|
|
switch (*fmt_it)
|
|
{
|
|
case LOG4CPLUS_TEXT ('q'):
|
|
{
|
|
if (! q_str_valid)
|
|
{
|
|
build_q_value (q_str);
|
|
q_str_valid = true;
|
|
}
|
|
ret.append (q_str);
|
|
state = TEXT;
|
|
}
|
|
break;
|
|
|
|
case LOG4CPLUS_TEXT ('Q'):
|
|
{
|
|
if (! uc_q_str_valid)
|
|
{
|
|
build_uc_q_value (uc_q_str);
|
|
uc_q_str_valid = true;
|
|
}
|
|
ret.append (uc_q_str);
|
|
state = TEXT;
|
|
}
|
|
break;
|
|
|
|
// Windows do not support %s format specifier
|
|
// (seconds since epoch).
|
|
case LOG4CPLUS_TEXT ('s'):
|
|
{
|
|
if (! s_str_valid)
|
|
{
|
|
convertIntegerToString (s_str, tv_sec);
|
|
s_str_valid = true;
|
|
}
|
|
ret.append (s_str);
|
|
state = TEXT;
|
|
}
|
|
break;
|
|
|
|
default:
|
|
{
|
|
ret.push_back (LOG4CPLUS_TEXT ('%'));
|
|
ret.push_back (*fmt_it);
|
|
state = TEXT;
|
|
}
|
|
}
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
|
|
// Finally call strftime/wcsftime to format the rest of the string.
|
|
|
|
ret.swap (fmt);
|
|
size_t buffer_size = fmt.size () + 1;
|
|
std::vector<tchar> buffer;
|
|
size_t len;
|
|
|
|
// Limit how far can the buffer grow. This is necessary so that we
|
|
// catch bad format string. Some implementations of strftime() signal
|
|
// both too small buffer and invalid format string by returning 0
|
|
// without changing errno.
|
|
size_t const buffer_size_max
|
|
= (std::max) (static_cast<size_t>(1024), buffer_size * 16);
|
|
|
|
do
|
|
{
|
|
buffer.resize (buffer_size);
|
|
errno = 0;
|
|
#ifdef UNICODE
|
|
len = helpers::wcsftime(&buffer[0], buffer_size, fmt.c_str(), &time);
|
|
#else
|
|
len = helpers::strftime(&buffer[0], buffer_size, fmt.c_str(), &time);
|
|
#endif
|
|
if (len == 0)
|
|
{
|
|
int const eno = errno;
|
|
buffer_size *= 2;
|
|
if (buffer_size > buffer_size_max)
|
|
{
|
|
log4cplus::tstring msg (
|
|
LOG4CPLUS_TEXT ("Error in strftime(): "));
|
|
msg += convertIntegerToString (eno);
|
|
LogLog::getLogLog ()->error (msg);
|
|
throw std::runtime_error (LOG4CPLUS_TSTRING_TO_STRING (msg));
|
|
}
|
|
}
|
|
}
|
|
while (len == 0);
|
|
ret.assign (buffer.begin (), buffer.begin () + len);
|
|
|
|
return ret;
|
|
}
|
|
|
|
|
|
Time&
|
|
Time::operator+=(const Time& rhs)
|
|
{
|
|
tv_sec += rhs.tv_sec;
|
|
tv_usec += rhs.tv_usec;
|
|
|
|
if(tv_usec > ONE_SEC_IN_USEC) {
|
|
++tv_sec;
|
|
tv_usec -= ONE_SEC_IN_USEC;
|
|
}
|
|
|
|
return *this;
|
|
}
|
|
|
|
|
|
Time&
|
|
Time::operator-=(const Time& rhs)
|
|
{
|
|
tv_sec -= rhs.tv_sec;
|
|
tv_usec -= rhs.tv_usec;
|
|
|
|
if(tv_usec < 0) {
|
|
--tv_sec;
|
|
tv_usec += ONE_SEC_IN_USEC;
|
|
}
|
|
|
|
return *this;
|
|
}
|
|
|
|
|
|
Time&
|
|
Time::operator/=(long rhs)
|
|
{
|
|
long rem_secs = static_cast<long>(tv_sec % rhs);
|
|
tv_sec /= rhs;
|
|
|
|
tv_usec /= rhs;
|
|
tv_usec += static_cast<long>((rem_secs * ONE_SEC_IN_USEC) / rhs);
|
|
|
|
return *this;
|
|
}
|
|
|
|
|
|
Time&
|
|
Time::operator*=(long rhs)
|
|
{
|
|
long new_usec = tv_usec * rhs;
|
|
long overflow_sec = new_usec / ONE_SEC_IN_USEC;
|
|
tv_usec = new_usec % ONE_SEC_IN_USEC;
|
|
|
|
tv_sec *= rhs;
|
|
tv_sec += overflow_sec;
|
|
|
|
return *this;
|
|
}
|
|
|
|
|
|
//////////////////////////////////////////////////////////////////////////////
|
|
// Time globals
|
|
//////////////////////////////////////////////////////////////////////////////
|
|
|
|
|
|
const Time
|
|
operator+(const Time& lhs, const Time& rhs)
|
|
{
|
|
return Time(lhs) += rhs;
|
|
}
|
|
|
|
|
|
const Time
|
|
operator-(const Time& lhs, const Time& rhs)
|
|
{
|
|
return Time(lhs) -= rhs;
|
|
}
|
|
|
|
|
|
const Time
|
|
operator/(const Time& lhs, long rhs)
|
|
{
|
|
return Time(lhs) /= rhs;
|
|
}
|
|
|
|
|
|
const Time
|
|
operator*(const Time& lhs, long rhs)
|
|
{
|
|
return Time(lhs) *= rhs;
|
|
}
|
|
|
|
|
|
bool
|
|
operator<(const Time& lhs, const Time& rhs)
|
|
{
|
|
return ( (lhs.sec() < rhs.sec())
|
|
|| ( (lhs.sec() == rhs.sec())
|
|
&& (lhs.usec() < rhs.usec())) );
|
|
}
|
|
|
|
|
|
bool
|
|
operator<=(const Time& lhs, const Time& rhs)
|
|
{
|
|
return ((lhs < rhs) || (lhs == rhs));
|
|
}
|
|
|
|
|
|
bool
|
|
operator>(const Time& lhs, const Time& rhs)
|
|
{
|
|
return ( (lhs.sec() > rhs.sec())
|
|
|| ( (lhs.sec() == rhs.sec())
|
|
&& (lhs.usec() > rhs.usec())) );
|
|
}
|
|
|
|
|
|
bool
|
|
operator>=(const Time& lhs, const Time& rhs)
|
|
{
|
|
return ((lhs > rhs) || (lhs == rhs));
|
|
}
|
|
|
|
|
|
bool
|
|
operator==(const Time& lhs, const Time& rhs)
|
|
{
|
|
return ( lhs.sec() == rhs.sec()
|
|
&& lhs.usec() == rhs.usec());
|
|
}
|
|
|
|
|
|
bool
|
|
operator!=(const Time& lhs, const Time& rhs)
|
|
{
|
|
return !(lhs == rhs);
|
|
}
|
|
|
|
|
|
} } // namespace log4cplus { namespace helpers {
|