9b6bf426b9
Also rename array_inline.h to array_data-inl.h
191 linhas
7.3 KiB
C++
191 linhas
7.3 KiB
C++
/*
|
|
+----------------------------------------------------------------------+
|
|
| HipHop for PHP |
|
|
+----------------------------------------------------------------------+
|
|
| Copyright (c) 2010-2013 Facebook, Inc. (http://www.facebook.com) |
|
|
+----------------------------------------------------------------------+
|
|
| This source file is subject to version 3.01 of the PHP license, |
|
|
| that is bundled with this package in the file LICENSE, and is |
|
|
| available through the world-wide-web at the following url: |
|
|
| http://www.php.net/license/3_01.txt |
|
|
| If you did not receive a copy of the PHP license and are unable to |
|
|
| obtain it through the world-wide-web, please send a note to |
|
|
| license@php.net so we can mail you a copy immediately. |
|
|
+----------------------------------------------------------------------+
|
|
*/
|
|
|
|
#include "hphp/runtime/base/util/extended_logger.h"
|
|
#include "hphp/runtime/base/execution_context.h"
|
|
#include "hphp/runtime/base/runtime_option.h"
|
|
#include "hphp/runtime/base/array_iterator.h"
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
|
|
#define IMPLEMENT_LOGLEVEL(LOGLEVEL) \
|
|
void ExtendedLogger::LOGLEVEL(const char *fmt, ...) { \
|
|
if (LogLevel < Log ## LOGLEVEL) return; \
|
|
if (RuntimeOption::InjectedStackTrace && \
|
|
!ExtendedLogger::EnabledByDefault) { \
|
|
Array bt = g_vmContext->debugBacktrace(); \
|
|
if (!bt.empty()) { \
|
|
va_list ap; va_start(ap, fmt); \
|
|
Logger::LogEscapeMore(Log ## LOGLEVEL, fmt, ap); \
|
|
va_end(ap); \
|
|
Log(Log ## LOGLEVEL, bt); \
|
|
return; \
|
|
} \
|
|
} \
|
|
va_list ap; va_start(ap, fmt); \
|
|
Logger::Log(Log ## LOGLEVEL, fmt, ap); \
|
|
va_end(ap); \
|
|
} \
|
|
void ExtendedLogger::LOGLEVEL(const std::string &msg) { \
|
|
if (LogLevel < Log ## LOGLEVEL) return; \
|
|
if (RuntimeOption::InjectedStackTrace && \
|
|
!ExtendedLogger::EnabledByDefault) { \
|
|
Array bt = g_vmContext->debugBacktrace(); \
|
|
if (!bt.empty()) { \
|
|
Logger::Log(Log ## LOGLEVEL, msg, nullptr, true, true); \
|
|
Log(Log ## LOGLEVEL, bt); \
|
|
return; \
|
|
} \
|
|
} \
|
|
Logger::Log(Log ## LOGLEVEL, msg, nullptr, true); \
|
|
}
|
|
|
|
namespace HPHP {
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
|
|
bool ExtendedLogger::EnabledByDefault = false;
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
|
|
IMPLEMENT_LOGLEVEL(Error);
|
|
IMPLEMENT_LOGLEVEL(Warning);
|
|
IMPLEMENT_LOGLEVEL(Info);
|
|
IMPLEMENT_LOGLEVEL(Verbose);
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
|
|
void ExtendedLogger::log(LogLevelType level, const char *type,
|
|
const Exception &e, const char *file /* = NULL */,
|
|
int line /* = 0 */) {
|
|
if (!IsEnabled()) return;
|
|
Logger::log(level, type, e, file, line);
|
|
|
|
if (RuntimeOption::InjectedStackTrace) {
|
|
const ExtendedException *ee = dynamic_cast<const ExtendedException *>(&e);
|
|
if (ee) {
|
|
Log(level, ee->getBackTrace());
|
|
}
|
|
}
|
|
}
|
|
|
|
void ExtendedLogger::log(LogLevelType level, const std::string &msg,
|
|
const StackTrace *stackTrace,
|
|
bool escape /* = true */,
|
|
bool escapeMore /* = false */) {
|
|
if (RuntimeOption::InjectedStackTrace) {
|
|
Array bt = g_vmContext->debugBacktrace();
|
|
if (!bt.empty()) {
|
|
Logger::log(level, msg, stackTrace, escape, escape);
|
|
Log(level, bt, escape, escapeMore);
|
|
return;
|
|
}
|
|
}
|
|
Logger::log(level, msg, stackTrace, escape, escapeMore);
|
|
}
|
|
|
|
void ExtendedLogger::Log(LogLevelType level, CArrRef stackTrace,
|
|
bool escape /* = true */,
|
|
bool escapeMore /* = false */) {
|
|
assert(!escapeMore || escape);
|
|
ThreadData *threadData = s_threadData.get();
|
|
if (++threadData->message > MaxMessagesPerRequest &&
|
|
MaxMessagesPerRequest >= 0) {
|
|
return;
|
|
}
|
|
|
|
if (stackTrace.isNull()) return;
|
|
|
|
if (UseLogFile) {
|
|
FILE *f = Output ? Output : GetStandardOut(level);
|
|
PrintStackTrace(f, stackTrace, escape, escapeMore);
|
|
|
|
FILE *tf = threadData->log;
|
|
if (tf) {
|
|
PrintStackTrace(tf, stackTrace, escape, escapeMore);
|
|
}
|
|
}
|
|
}
|
|
|
|
const StaticString
|
|
s_class("class"),
|
|
s_function("function"),
|
|
s_file("file"),
|
|
s_type("type"),
|
|
s_line("line");
|
|
|
|
void ExtendedLogger::PrintStackTrace(FILE *f, CArrRef stackTrace,
|
|
bool escape /* = false */,
|
|
bool escapeMore /* = false */) {
|
|
int i = 0;
|
|
for (ArrayIter it(stackTrace); it; ++it, ++i) {
|
|
if (i > 0) {
|
|
fprintf(f, "%s", escape ? "\\n" : "\n");
|
|
}
|
|
Array frame = it.second().toArray();
|
|
fprintf(f, " #%d ", i);
|
|
if (frame.exists(s_function)) {
|
|
if (frame.exists(s_class)) {
|
|
fprintf(f, "%s%s%s(), called ", frame[s_class].toString().c_str(),
|
|
frame[s_type].toString().c_str(),
|
|
frame[s_function].toString().c_str());
|
|
} else {
|
|
fprintf(f, "%s(), called ", frame[s_function].toString().c_str());
|
|
}
|
|
}
|
|
fprintf(f, "at [%s:%" PRId64 "]", frame[s_file].toString().c_str(),
|
|
frame[s_line].toInt64());
|
|
}
|
|
fprintf(f, escapeMore ? "\\n" : "\n");
|
|
fflush(f);
|
|
}
|
|
|
|
std::string ExtendedLogger::StringOfFrame(CArrRef frame, int i, bool escape) {
|
|
std::ostringstream ss;
|
|
|
|
if (i > 0) {
|
|
ss << (escape ? "\\n" : "\n");
|
|
}
|
|
ss << " #" << i << " ";
|
|
if (frame.exists(s_function)) {
|
|
if (frame.exists(s_class)) {
|
|
ss << frame[s_class].toString().c_str()
|
|
<< frame[s_type].toString().c_str()
|
|
<< frame[s_function].toString().c_str()
|
|
<< "(), called ";
|
|
} else {
|
|
ss << frame[s_function].toString().c_str()
|
|
<< "(), called ";
|
|
}
|
|
}
|
|
ss << "at [" << frame[s_file].toString().c_str()
|
|
<< ":" << frame[s_line].toInt64() << "]";
|
|
|
|
return ss.str();
|
|
}
|
|
|
|
std::string ExtendedLogger::StringOfStackTrace(CArrRef stackTrace) {
|
|
std::string buf;
|
|
int i = 0;
|
|
for (ArrayIter it(stackTrace); it; ++it, ++i) {
|
|
buf += StringOfFrame(it.second().toArray(), i);
|
|
}
|
|
buf += "\n";
|
|
return buf;
|
|
}
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
}
|