Commit d0531421 authored by Shinichiro Hamaji's avatar Shinichiro Hamaji Committed by GitHub

Merge pull request #226 from sergiud/zero-allocation-log-failure

Zero allocation fix
parents 2a6df662 2fe6508e
...@@ -192,6 +192,11 @@ using namespace Outer::Inner;; ...@@ -192,6 +192,11 @@ using namespace Outer::Inner;;
int main() { return i; } int main() { return i; }
" HAVE_NAMESPACES) " HAVE_NAMESPACES)
check_cxx_source_compiles ("
__thread int tls;
int main() { }
" HAVE_GCC_TLS)
check_cxx_source_compiles (" check_cxx_source_compiles ("
__declspec(thread) int tls; __declspec(thread) int tls;
int main() { } int main() { }
...@@ -202,19 +207,15 @@ thread_local int tls; ...@@ -202,19 +207,15 @@ thread_local int tls;
int main() { } int main() { }
" HAVE_CXX11_TLS) " HAVE_CXX11_TLS)
check_cxx_source_compiles ("
__attribute__((thread)) int tls;
int main() { }
" HAVE_CYGWIN_TLS)
if (WITH_TLS) if (WITH_TLS)
if (HAVE_CYGWIN_TLS) # Cygwin does not support the thread attribute. Don't bother.
set (GLOG_THREAD_LOCAL_STORAGE "__attribute__((thread))") if (HAVE_GCC_TLS)
set (GLOG_THREAD_LOCAL_STORAGE "__thread")
elseif (HAVE_MSVC_TLS) elseif (HAVE_MSVC_TLS)
set (GLOG_THREAD_LOCAL_STORAGE "__declspec(thread)") set (GLOG_THREAD_LOCAL_STORAGE "__declspec(thread)")
elseif (HAVE_CXX11_TLS) elseif (HAVE_CXX11_TLS)
set (GLOG_THREAD_LOCAL_STORAGE thread_local) set (GLOG_THREAD_LOCAL_STORAGE thread_local)
endif (HAVE_CYGWIN_TLS) endif (HAVE_GCC_TLS)
endif (WITH_TLS) endif (WITH_TLS)
set (_PC_FIELDS set (_PC_FIELDS
......
...@@ -1109,11 +1109,6 @@ class GOOGLE_GLOG_DLL_DECL LogStreamBuf : public std::streambuf { ...@@ -1109,11 +1109,6 @@ class GOOGLE_GLOG_DLL_DECL LogStreamBuf : public std::streambuf {
setp(buf, buf + len - 2); setp(buf, buf + len - 2);
} }
// Resets the buffer. Useful if we reuse it by means of TLS.
void reset() {
setp(pbase(), epptr());
}
// This effectively ignores overflow. // This effectively ignores overflow.
virtual int_type overflow(int_type ch) { virtual int_type overflow(int_type ch) {
return ch; return ch;
...@@ -1176,7 +1171,6 @@ public: ...@@ -1176,7 +1171,6 @@ public:
size_t pcount() const { return streambuf_.pcount(); } size_t pcount() const { return streambuf_.pcount(); }
char* pbase() const { return streambuf_.pbase(); } char* pbase() const { return streambuf_.pbase(); }
char* str() const { return pbase(); } char* str() const { return pbase(); }
void reset() { streambuf_.reset(); }
private: private:
LogStream(const LogStream&); LogStream(const LogStream&);
......
...@@ -331,7 +331,6 @@ const size_t LogMessage::kMaxLogMessageLen = 30000; ...@@ -331,7 +331,6 @@ const size_t LogMessage::kMaxLogMessageLen = 30000;
struct LogMessage::LogMessageData { struct LogMessage::LogMessageData {
LogMessageData(); LogMessageData();
void reset();
int preserved_errno_; // preserved errno int preserved_errno_; // preserved errno
// Buffer space; contains complete message text. // Buffer space; contains complete message text.
...@@ -1151,17 +1150,13 @@ static LogMessage::LogMessageData fatal_msg_data_shared; ...@@ -1151,17 +1150,13 @@ static LogMessage::LogMessageData fatal_msg_data_shared;
// LogMessageData object exists (in this case glog makes zero heap memory // LogMessageData object exists (in this case glog makes zero heap memory
// allocations). // allocations).
static GLOG_THREAD_LOCAL_STORAGE bool thread_data_available = true; static GLOG_THREAD_LOCAL_STORAGE bool thread_data_available = true;
static GLOG_THREAD_LOCAL_STORAGE LogMessage::LogMessageData thread_msg_data; static GLOG_THREAD_LOCAL_STORAGE char thread_msg_data[sizeof(LogMessage::LogMessageData)];
#endif // defined(GLOG_THREAD_LOCAL_STORAGE) #endif // defined(GLOG_THREAD_LOCAL_STORAGE)
LogMessage::LogMessageData::LogMessageData() LogMessage::LogMessageData::LogMessageData()
: stream_(message_text_, LogMessage::kMaxLogMessageLen, 0) { : stream_(message_text_, LogMessage::kMaxLogMessageLen, 0) {
} }
void LogMessage::LogMessageData::reset() {
stream_.reset();
}
LogMessage::LogMessage(const char* file, int line, LogSeverity severity, LogMessage::LogMessage(const char* file, int line, LogSeverity severity,
int ctr, void (LogMessage::*send_method)()) int ctr, void (LogMessage::*send_method)())
: allocated_(NULL) { : allocated_(NULL) {
...@@ -1218,10 +1213,7 @@ void LogMessage::Init(const char* file, ...@@ -1218,10 +1213,7 @@ void LogMessage::Init(const char* file,
// No need for locking, because this is thread local. // No need for locking, because this is thread local.
if (thread_data_available) { if (thread_data_available) {
thread_data_available = false; thread_data_available = false;
data_ = &thread_msg_data; data_ = new (&thread_msg_data) LogMessageData;
// Make sure to clear log data since it may have been used and filled with
// data. We do not want to append the new message to the previous one.
data_->reset();
} else { } else {
allocated_ = new LogMessageData(); allocated_ = new LogMessageData();
data_ = allocated_; data_ = allocated_;
...@@ -1299,10 +1291,16 @@ void LogMessage::Init(const char* file, ...@@ -1299,10 +1291,16 @@ void LogMessage::Init(const char* file,
LogMessage::~LogMessage() { LogMessage::~LogMessage() {
Flush(); Flush();
#ifdef GLOG_THREAD_LOCAL_STORAGE #ifdef GLOG_THREAD_LOCAL_STORAGE
if (data_ == &thread_msg_data) if (data_ == static_cast<void*>(thread_msg_data)) {
data_->~LogMessageData();
thread_data_available = true; thread_data_available = true;
#endif // defined(GLOG_THREAD_LOCAL_STORAGE) }
else {
delete allocated_;
}
#else // !defined(GLOG_THREAD_LOCAL_STORAGE)
delete allocated_; delete allocated_;
#endif // defined(GLOG_THREAD_LOCAL_STORAGE)
} }
int LogMessage::preserved_errno() const { int LogMessage::preserved_errno() const {
......
...@@ -272,7 +272,12 @@ void TestLogging(bool check_counts) { ...@@ -272,7 +272,12 @@ void TestLogging(bool check_counts) {
LOG(ERROR) << string("foo") << ' '<< j << ' ' << setw(10) << j << " " LOG(ERROR) << string("foo") << ' '<< j << ' ' << setw(10) << j << " "
<< setw(1) << hex << j; << setw(1) << hex << j;
LOG(ERROR) << (&LOG(ERROR) && 0) << " nested LOG"; {
google::LogMessage outer(__FILE__, __LINE__, google::ERROR);
outer.stream() << "outer";
LOG(ERROR) << "inner";
}
LogMessage("foo", LogMessage::kNoLogPrefix, GLOG_INFO).stream() << "no prefix"; LogMessage("foo", LogMessage::kNoLogPrefix, GLOG_INFO).stream() << "no prefix";
......
...@@ -74,8 +74,8 @@ WDATE TIME__ THREADID logging_unittest.cc:LINE] log_if this ...@@ -74,8 +74,8 @@ WDATE TIME__ THREADID logging_unittest.cc:LINE] log_if this
IDATE TIME__ THREADID logging_unittest.cc:LINE] array IDATE TIME__ THREADID logging_unittest.cc:LINE] array
IDATE TIME__ THREADID logging_unittest.cc:LINE] const array IDATE TIME__ THREADID logging_unittest.cc:LINE] const array
EDATE TIME__ THREADID logging_unittest.cc:LINE] foo 1000 0000001000 3e8 EDATE TIME__ THREADID logging_unittest.cc:LINE] foo 1000 0000001000 3e8
EDATE TIME__ THREADID logging_unittest.cc:LINE] 0 nested LOG EDATE TIME__ THREADID logging_unittest.cc:LINE] inner
EDATE TIME__ THREADID logging_unittest.cc:LINE] EDATE TIME__ THREADID logging_unittest.cc:LINE] outer
no prefix no prefix
IDATE TIME__ THREADID logging_unittest.cc:LINE] RAW: foo bar 10 3.400000 IDATE TIME__ THREADID logging_unittest.cc:LINE] RAW: foo bar 10 3.400000
WDATE TIME__ THREADID logging_unittest.cc:LINE] RAW: array WDATE TIME__ THREADID logging_unittest.cc:LINE] RAW: array
......
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment