forked from eden-emu/eden
		
	
		
			
				
	
	
		
			364 lines
		
	
	
	
		
			12 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			364 lines
		
	
	
	
		
			12 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| // Copyright 2018 yuzu emulator team
 | |
| // Licensed under GPLv2 or any later version
 | |
| // Refer to the license.txt file included.
 | |
| 
 | |
| #include <sstream>
 | |
| #include <string>
 | |
| 
 | |
| #include <optional>
 | |
| #include <unordered_map>
 | |
| #include <boost/container_hash/hash.hpp>
 | |
| #include "common/logging/log.h"
 | |
| #include "common/scope_exit.h"
 | |
| #include "core/core.h"
 | |
| #include "core/hle/ipc_helpers.h"
 | |
| #include "core/hle/service/lm/lm.h"
 | |
| #include "core/hle/service/service.h"
 | |
| #include "core/memory.h"
 | |
| 
 | |
| namespace Service::LM {
 | |
| enum class LogSeverity : u8 {
 | |
|     Trace = 0,
 | |
|     Info = 1,
 | |
|     Warning = 2,
 | |
|     Error = 3,
 | |
|     Fatal = 4,
 | |
| };
 | |
| 
 | |
| // To keep flags out of hashing as well as the payload size
 | |
| struct LogPacketHeaderEntry {
 | |
|     u64_le pid{};
 | |
|     u64_le tid{};
 | |
|     LogSeverity severity{};
 | |
|     u8 verbosity{};
 | |
| 
 | |
|     auto operator<=>(const LogPacketHeaderEntry&) const = default;
 | |
| };
 | |
| } // namespace Service::LM
 | |
| 
 | |
| namespace std {
 | |
| template <>
 | |
| struct hash<Service::LM::LogPacketHeaderEntry> {
 | |
|     std::size_t operator()(const Service::LM::LogPacketHeaderEntry& k) const noexcept {
 | |
|         std::size_t seed{};
 | |
|         boost::hash_combine(seed, k.pid);
 | |
|         boost::hash_combine(seed, k.tid);
 | |
|         boost::hash_combine(seed, k.severity);
 | |
|         boost::hash_combine(seed, k.verbosity);
 | |
|         return seed;
 | |
|     };
 | |
| };
 | |
| } // namespace std
 | |
| 
 | |
| namespace Service::LM {
 | |
| 
 | |
| enum class LogDestination : u32 {
 | |
|     TargetManager = 1 << 0,
 | |
|     Uart = 1 << 1,
 | |
|     UartSleep = 1 << 2,
 | |
|     All = 0xffff,
 | |
| };
 | |
| DECLARE_ENUM_FLAG_OPERATORS(LogDestination);
 | |
| 
 | |
| enum class LogPacketFlags : u8 {
 | |
|     Head = 1 << 0,
 | |
|     Tail = 1 << 1,
 | |
|     LittleEndian = 1 << 2,
 | |
| };
 | |
| DECLARE_ENUM_FLAG_OPERATORS(LogPacketFlags);
 | |
| 
 | |
| class ILogger final : public ServiceFramework<ILogger> {
 | |
| public:
 | |
|     explicit ILogger(Core::System& system_) : ServiceFramework{system_, "ILogger"} {
 | |
|         static const FunctionInfo functions[] = {
 | |
|             {0, &ILogger::Log, "Log"},
 | |
|             {1, &ILogger::SetDestination, "SetDestination"},
 | |
|         };
 | |
|         RegisterHandlers(functions);
 | |
|     }
 | |
| 
 | |
| private:
 | |
|     void Log(Kernel::HLERequestContext& ctx) {
 | |
|         std::size_t offset{};
 | |
|         const auto data = ctx.ReadBuffer();
 | |
| 
 | |
|         // This function only succeeds - Get that out of the way
 | |
|         IPC::ResponseBuilder rb{ctx, 2};
 | |
|         rb.Push(RESULT_SUCCESS);
 | |
| 
 | |
|         if (data.size() < sizeof(LogPacketHeader)) {
 | |
|             LOG_ERROR(Service_LM, "Data size is too small for header! size={}", data.size());
 | |
|             return;
 | |
|         }
 | |
| 
 | |
|         LogPacketHeader header{};
 | |
|         std::memcpy(&header, data.data(), sizeof(LogPacketHeader));
 | |
|         offset += sizeof(LogPacketHeader);
 | |
| 
 | |
|         LogPacketHeaderEntry entry{
 | |
|             .pid = header.pid,
 | |
|             .tid = header.tid,
 | |
|             .severity = header.severity,
 | |
|             .verbosity = header.verbosity,
 | |
|         };
 | |
| 
 | |
|         if (True(header.flags & LogPacketFlags::Head)) {
 | |
|             std::vector<u8> tmp(data.size() - sizeof(LogPacketHeader));
 | |
|             std::memcpy(tmp.data(), data.data() + offset, tmp.size());
 | |
|             entries[entry] = std::move(tmp);
 | |
|         } else {
 | |
|             // Append to existing entry
 | |
|             if (!entries.contains(entry)) {
 | |
|                 LOG_ERROR(Service_LM, "Log entry does not exist!");
 | |
|                 return;
 | |
|             }
 | |
|             std::vector<u8> tmp(data.size() - sizeof(LogPacketHeader));
 | |
| 
 | |
|             auto& existing_entry = entries[entry];
 | |
|             const auto base = existing_entry.size();
 | |
|             existing_entry.resize(base + (data.size() - sizeof(LogPacketHeader)));
 | |
|             std::memcpy(existing_entry.data() + base, data.data() + offset,
 | |
|                         (data.size() - sizeof(LogPacketHeader)));
 | |
|         }
 | |
| 
 | |
|         if (True(header.flags & LogPacketFlags::Tail)) {
 | |
|             auto it = entries.find(entry);
 | |
|             if (it == entries.end()) {
 | |
|                 LOG_ERROR(Service_LM, "Log entry does not exist!");
 | |
|                 return;
 | |
|             }
 | |
|             ParseLog(it->first, it->second);
 | |
|             entries.erase(it);
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     void SetDestination(Kernel::HLERequestContext& ctx) {
 | |
|         IPC::RequestParser rp{ctx};
 | |
|         const auto log_destination = rp.PopEnum<LogDestination>();
 | |
| 
 | |
|         LOG_DEBUG(Service_LM, "called, destination={}", DestinationToString(log_destination));
 | |
|         destination = log_destination;
 | |
| 
 | |
|         IPC::ResponseBuilder rb{ctx, 2};
 | |
|         rb.Push(RESULT_SUCCESS);
 | |
|     }
 | |
| 
 | |
|     u32 ReadLeb128(const std::vector<u8>& data, std::size_t& offset) {
 | |
|         u32 result{};
 | |
|         u32 shift{};
 | |
|         do {
 | |
|             result |= (data[offset] & 0x7f) << shift;
 | |
|             shift += 7;
 | |
|             offset++;
 | |
|             if (offset >= data.size()) {
 | |
|                 break;
 | |
|             }
 | |
|         } while ((data[offset] & 0x80) != 0);
 | |
|         return result;
 | |
|     }
 | |
| 
 | |
|     std::optional<std::string> ReadString(const std::vector<u8>& data, std::size_t& offset,
 | |
|                                           std::size_t length) {
 | |
|         if (length == 0) {
 | |
|             return std::nullopt;
 | |
|         }
 | |
|         std::string output(length, '\0');
 | |
|         std::memcpy(output.data(), data.data() + offset, length);
 | |
|         offset += length;
 | |
|         return output;
 | |
|     }
 | |
| 
 | |
|     u32_le ReadAsU32(const std::vector<u8>& data, std::size_t& offset, std::size_t length) {
 | |
|         ASSERT(length == sizeof(u32));
 | |
|         u32_le output{};
 | |
|         std::memcpy(&output, data.data() + offset, sizeof(u32));
 | |
|         offset += length;
 | |
|         return output;
 | |
|     }
 | |
| 
 | |
|     u64_le ReadAsU64(const std::vector<u8>& data, std::size_t& offset, std::size_t length) {
 | |
|         ASSERT(length == sizeof(u64));
 | |
|         u64_le output{};
 | |
|         std::memcpy(&output, data.data() + offset, sizeof(u64));
 | |
|         offset += length;
 | |
|         return output;
 | |
|     }
 | |
| 
 | |
|     void ParseLog(const LogPacketHeaderEntry entry, const std::vector<u8>& log_data) {
 | |
|         // Possible entries
 | |
|         std::optional<std::string> text_log;
 | |
|         std::optional<u32> line_number;
 | |
|         std::optional<std::string> file_name;
 | |
|         std::optional<std::string> function_name;
 | |
|         std::optional<std::string> module_name;
 | |
|         std::optional<std::string> thread_name;
 | |
|         std::optional<u64> log_pack_drop_count;
 | |
|         std::optional<s64> user_system_clock;
 | |
|         std::optional<std::string> process_name;
 | |
| 
 | |
|         std::size_t offset{};
 | |
|         while (offset < log_data.size()) {
 | |
|             const auto key = static_cast<LogDataChunkKey>(ReadLeb128(log_data, offset));
 | |
|             const auto chunk_size = ReadLeb128(log_data, offset);
 | |
| 
 | |
|             switch (key) {
 | |
|             case LogDataChunkKey::LogSessionBegin:
 | |
|             case LogDataChunkKey::LogSessionEnd:
 | |
|                 break;
 | |
|             case LogDataChunkKey::TextLog:
 | |
|                 text_log = ReadString(log_data, offset, chunk_size);
 | |
|                 break;
 | |
|             case LogDataChunkKey::LineNumber:
 | |
|                 line_number = ReadAsU32(log_data, offset, chunk_size);
 | |
|                 break;
 | |
|             case LogDataChunkKey::FileName:
 | |
|                 file_name = ReadString(log_data, offset, chunk_size);
 | |
|                 break;
 | |
|             case LogDataChunkKey::FunctionName:
 | |
|                 function_name = ReadString(log_data, offset, chunk_size);
 | |
|                 break;
 | |
|             case LogDataChunkKey::ModuleName:
 | |
|                 module_name = ReadString(log_data, offset, chunk_size);
 | |
|                 break;
 | |
|             case LogDataChunkKey::ThreadName:
 | |
|                 thread_name = ReadString(log_data, offset, chunk_size);
 | |
|                 break;
 | |
|             case LogDataChunkKey::LogPacketDropCount:
 | |
|                 log_pack_drop_count = ReadAsU64(log_data, offset, chunk_size);
 | |
|                 break;
 | |
|             case LogDataChunkKey::UserSystemClock:
 | |
|                 user_system_clock = ReadAsU64(log_data, offset, chunk_size);
 | |
|                 break;
 | |
|             case LogDataChunkKey::ProcessName:
 | |
|                 process_name = ReadString(log_data, offset, chunk_size);
 | |
|                 break;
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         std::string output_log{};
 | |
|         if (process_name) {
 | |
|             output_log += fmt::format("Process: {}\n", *process_name);
 | |
|         }
 | |
|         if (module_name) {
 | |
|             output_log += fmt::format("Module: {}\n", *module_name);
 | |
|         }
 | |
|         if (file_name) {
 | |
|             output_log += fmt::format("File: {}\n", *file_name);
 | |
|         }
 | |
|         if (function_name) {
 | |
|             output_log += fmt::format("Function: {}\n", *function_name);
 | |
|         }
 | |
|         if (line_number && *line_number != 0) {
 | |
|             output_log += fmt::format("Line: {}\n", *line_number);
 | |
|         }
 | |
|         output_log += fmt::format("ProcessID: {}\n", entry.pid);
 | |
|         output_log += fmt::format("ThreadID: {}\n", entry.tid);
 | |
| 
 | |
|         if (text_log) {
 | |
|             output_log += fmt::format("Log Text: {}\n", *text_log);
 | |
|         }
 | |
| 
 | |
|         switch (entry.severity) {
 | |
|         case LogSeverity::Trace:
 | |
|             LOG_DEBUG(Service_LM, "LogManager DEBUG ({}):\n{}", DestinationToString(destination),
 | |
|                       output_log);
 | |
|             break;
 | |
|         case LogSeverity::Info:
 | |
|             LOG_INFO(Service_LM, "LogManager INFO ({}):\n{}", DestinationToString(destination),
 | |
|                      output_log);
 | |
|             break;
 | |
|         case LogSeverity::Warning:
 | |
|             LOG_WARNING(Service_LM, "LogManager WARNING ({}):\n{}",
 | |
|                         DestinationToString(destination), output_log);
 | |
|             break;
 | |
|         case LogSeverity::Error:
 | |
|             LOG_ERROR(Service_LM, "LogManager ERROR ({}):\n{}", DestinationToString(destination),
 | |
|                       output_log);
 | |
|             break;
 | |
|         case LogSeverity::Fatal:
 | |
|             LOG_CRITICAL(Service_LM, "LogManager FATAL ({}):\n{}", DestinationToString(destination),
 | |
|                          output_log);
 | |
|             break;
 | |
|         default:
 | |
|             LOG_CRITICAL(Service_LM, "LogManager UNKNOWN ({}):\n{}",
 | |
|                          DestinationToString(destination), output_log);
 | |
|             break;
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     std::string DestinationToString(LogDestination destination) {
 | |
|         if (True(destination & LogDestination::All)) {
 | |
|             return "TargetManager | Uart | UartSleep";
 | |
|         }
 | |
|         std::string output{};
 | |
|         if (True(destination & LogDestination::TargetManager)) {
 | |
|             output += "| TargetManager";
 | |
|         }
 | |
|         if (True(destination & LogDestination::Uart)) {
 | |
|             output += "| Uart";
 | |
|         }
 | |
|         if (True(destination & LogDestination::UartSleep)) {
 | |
|             output += "| UartSleep";
 | |
|         }
 | |
|         if (output.length() > 0) {
 | |
|             return output.substr(2);
 | |
|         }
 | |
|         return "No Destination";
 | |
|     }
 | |
| 
 | |
|     enum class LogDataChunkKey : u32 {
 | |
|         LogSessionBegin = 0,
 | |
|         LogSessionEnd = 1,
 | |
|         TextLog = 2,
 | |
|         LineNumber = 3,
 | |
|         FileName = 4,
 | |
|         FunctionName = 5,
 | |
|         ModuleName = 6,
 | |
|         ThreadName = 7,
 | |
|         LogPacketDropCount = 8,
 | |
|         UserSystemClock = 9,
 | |
|         ProcessName = 10,
 | |
|     };
 | |
| 
 | |
|     struct LogPacketHeader {
 | |
|         u64_le pid{};
 | |
|         u64_le tid{};
 | |
|         LogPacketFlags flags{};
 | |
|         INSERT_PADDING_BYTES(1);
 | |
|         LogSeverity severity{};
 | |
|         u8 verbosity{};
 | |
|         u32_le payload_size{};
 | |
|     };
 | |
|     static_assert(sizeof(LogPacketHeader) == 0x18, "LogPacketHeader is an invalid size");
 | |
| 
 | |
|     std::unordered_map<LogPacketHeaderEntry, std::vector<u8>> entries{};
 | |
|     LogDestination destination{LogDestination::All};
 | |
| };
 | |
| 
 | |
| class LM final : public ServiceFramework<LM> {
 | |
| public:
 | |
|     explicit LM(Core::System& system_) : ServiceFramework{system_, "lm"} {
 | |
|         // clang-format off
 | |
|         static const FunctionInfo functions[] = {
 | |
|             {0, &LM::OpenLogger, "OpenLogger"},
 | |
|         };
 | |
|         // clang-format on
 | |
| 
 | |
|         RegisterHandlers(functions);
 | |
|     }
 | |
| 
 | |
| private:
 | |
|     void OpenLogger(Kernel::HLERequestContext& ctx) {
 | |
|         LOG_DEBUG(Service_LM, "called");
 | |
| 
 | |
|         IPC::ResponseBuilder rb{ctx, 2, 0, 1};
 | |
|         rb.Push(RESULT_SUCCESS);
 | |
|         rb.PushIpcInterface<ILogger>(system);
 | |
|     }
 | |
| };
 | |
| 
 | |
| void InstallInterfaces(Core::System& system) {
 | |
|     std::make_shared<LM>(system)->InstallAsService(system.ServiceManager());
 | |
| }
 | |
| 
 | |
| } // namespace Service::LM
 | 
