Service/lm: add support to multiline logs

This commit is contained in:
mailwl 2018-02-13 09:44:53 +03:00
parent ce8006e851
commit 55de13efcc

View File

@ -2,6 +2,7 @@
// Licensed under GPLv2 or any later version // Licensed under GPLv2 or any later version
// Refer to the license.txt file included. // Refer to the license.txt file included.
#include <sstream>
#include <string> #include <string>
#include "common/logging/log.h" #include "common/logging/log.h"
#include "core/hle/ipc_helpers.h" #include "core/hle/ipc_helpers.h"
@ -28,19 +29,28 @@ private:
IsHead = 1, IsHead = 1,
IsTail = 2, IsTail = 2,
}; };
enum Severity : u32_le {
Trace,
Info,
Warning,
Error,
Critical,
};
u64_le pid; u64_le pid;
u64_le threadContext; u64_le threadContext;
union { union {
BitField<0, 16, Flags> flags; BitField<0, 16, Flags> flags;
BitField<16, 8, u32_le> severity; BitField<16, 8, Severity> severity;
BitField<24, 8, u32_le> verbosity; BitField<24, 8, u32_le> verbosity;
}; };
u32_le payload_size; u32_le payload_size;
/// Returns true if this is part of a single log message bool IsHeadLog() const {
bool IsSingleMessage() const { return flags & Flags::IsHead;
return (flags & Flags::IsHead) && (flags & Flags::IsTail); }
bool IsTailLog() const {
return flags & Flags::IsTail;
} }
}; };
static_assert(sizeof(MessageHeader) == 0x18, "MessageHeader is incorrect size"); static_assert(sizeof(MessageHeader) == 0x18, "MessageHeader is incorrect size");
@ -57,7 +67,7 @@ private:
}; };
/** /**
* LM::Initialize service function * LM::Log service function
* Inputs: * Inputs:
* 0: 0x00000000 * 0: 0x00000000
* Outputs: * Outputs:
@ -75,9 +85,9 @@ private:
Memory::ReadBlock(addr, &header, sizeof(MessageHeader)); Memory::ReadBlock(addr, &header, sizeof(MessageHeader));
addr += sizeof(MessageHeader); addr += sizeof(MessageHeader);
if (!header.IsSingleMessage()) { if (header.IsHeadLog()) {
LOG_WARNING(Service_LM, "Multi message logs are unimplemeneted"); log_stream.str("");
return; log_stream.clear();
} }
// Parse out log metadata // Parse out log metadata
@ -85,7 +95,7 @@ private:
std::string message, filename, function; std::string message, filename, function;
while (addr < end_addr) { while (addr < end_addr) {
const Field field{static_cast<Field>(Memory::Read8(addr++))}; const Field field{static_cast<Field>(Memory::Read8(addr++))};
size_t length{Memory::Read8(addr++)}; const size_t length{Memory::Read8(addr++)};
if (static_cast<Field>(Memory::Read8(addr)) == Field::Skip) { if (static_cast<Field>(Memory::Read8(addr)) == Field::Skip) {
++addr; ++addr;
@ -110,28 +120,47 @@ private:
} }
// Empty log - nothing to do here // Empty log - nothing to do here
if (message.empty()) { if (log_stream.str().empty() && message.empty()) {
return; return;
} }
// Format a nicely printable string out of the log metadata // Format a nicely printable string out of the log metadata
std::string output; if (!filename.empty()) {
if (filename.size()) { log_stream << filename << ':';
output += filename + ':';
} }
if (function.size()) { if (!function.empty()) {
output += function + ':'; log_stream << function << ':';
} }
if (line) { if (line) {
output += std::to_string(line) + ':'; log_stream << std::to_string(line) << ':';
} }
if (output.length() > 0 && output.back() == ':') { if (log_stream.str().length() > 0 && log_stream.str().back() == ':') {
output += ' '; log_stream << ' ';
} }
output += message; log_stream << message;
LOG_INFO(Debug_Emulated, "%s", output.c_str()); if (header.IsTailLog()) {
switch (header.severity) {
case MessageHeader::Severity::Trace:
LOG_TRACE(Debug_Emulated, "%s", log_stream.str().c_str());
break;
case MessageHeader::Severity::Info:
LOG_INFO(Debug_Emulated, "%s", log_stream.str().c_str());
break;
case MessageHeader::Severity::Warning:
LOG_WARNING(Debug_Emulated, "%s", log_stream.str().c_str());
break;
case MessageHeader::Severity::Error:
LOG_ERROR(Debug_Emulated, "%s", log_stream.str().c_str());
break;
case MessageHeader::Severity::Critical:
LOG_CRITICAL(Debug_Emulated, "%s", log_stream.str().c_str());
break;
} }
}
}
std::ostringstream log_stream;
}; };
void InstallInterfaces(SM::ServiceManager& service_manager) { void InstallInterfaces(SM::ServiceManager& service_manager) {