// Copyright 2013 The Chromium Authors. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
// Only compile this file in debug build. This gives us one more level of
// protection that if the linker tries to link in strings/symbols appended to
// "DLOG() <<" in release build (which it shouldn't), we'll get "undefined
// reference" errors.
#if !defined(NDEBUG)
#include "media/cdm/ppapi/cdm_logging.h"
#include "base/basictypes.h"
#if defined(OS_WIN)
#include <io.h>
#include <windows.h>
#elif defined(OS_MACOSX)
#include <mach/mach.h>
#include <mach/mach_time.h>
#include <mach-o/dyld.h>
#elif defined(OS_POSIX)
#include <sys/syscall.h>
#include <time.h>
#endif
#if defined(OS_POSIX)
#include <errno.h>
#include <pthread.h>
#include <stdlib.h>
#include <stdio.h>
#include <string.h>
#include <unistd.h>
#endif
#include <iomanip>
#include <string>
namespace media {
namespace {
// Helper functions to wrap platform differences.
int32 CurrentProcessId() {
#if defined(OS_WIN)
return GetCurrentProcessId();
#elif defined(OS_POSIX)
return getpid();
#endif
}
int32 CurrentThreadId() {
// Pthreads doesn't have the concept of a thread ID, so we have to reach down
// into the kernel.
#if defined(OS_LINUX)
return syscall(__NR_gettid);
#elif defined(OS_ANDROID)
return gettid();
#elif defined(OS_SOLARIS)
return pthread_self();
#elif defined(OS_POSIX)
return reinterpret_cast<int64>(pthread_self());
#elif defined(OS_WIN)
return static_cast<int32>(::GetCurrentThreadId());
#endif
}
uint64 TickCount() {
#if defined(OS_WIN)
return GetTickCount();
#elif defined(OS_MACOSX)
return mach_absolute_time();
#elif defined(OS_POSIX)
struct timespec ts;
clock_gettime(CLOCK_MONOTONIC, &ts);
uint64 absolute_micro =
static_cast<int64>(ts.tv_sec) * 1000000 +
static_cast<int64>(ts.tv_nsec) / 1000;
return absolute_micro;
#endif
}
} // namespace
CdmLogMessage::CdmLogMessage(const char* file, int line) {
std::string filename(file);
size_t last_slash_pos = filename.find_last_of("\\/");
if (last_slash_pos != std::string::npos)
filename = filename.substr(last_slash_pos + 1);
stream_ << '[';
// Process and thread ID.
stream_ << CurrentProcessId() << ':';
stream_ << CurrentThreadId() << ':';
// Time and tick count.
time_t t = time(NULL);
struct tm local_time = {0};
#if _MSC_VER >= 1400
localtime_s(&local_time, &t);
#else
localtime_r(&t, &local_time);
#endif
struct tm* tm_time = &local_time;
stream_ << std::setfill('0')
<< std::setw(2) << 1 + tm_time->tm_mon
<< std::setw(2) << tm_time->tm_mday
<< '/'
<< std::setw(2) << tm_time->tm_hour
<< std::setw(2) << tm_time->tm_min
<< std::setw(2) << tm_time->tm_sec
<< ':';
stream_ << TickCount() << ':';
// File name.
stream_ << filename << "(" << line << ")] ";
}
CdmLogMessage::~CdmLogMessage() {
// Use std::cout explicitly for the line break. This limits the use of this
// class only to the definition of DLOG() (which also uses std::cout).
//
// This appends "std::endl" after all other messages appended to DLOG(),
// which relies on the C++ standard ISO/IEC 14882:1998(E) $12.2.3:
// "Temporary objects are destroyed as the last step in evaluating the
// full-expression (1.9) that (lexically) contains the point where they were
// created."
std::cout << std::endl;
}
} // namespace media
#endif // !defined(NDEBUG)