summaryrefslogtreecommitdiffstats
path: root/src/core/hle/service/lm/manager.h
blob: 544e636ba41749f82767c65a7f3fa1e3a80ea35f (plain) (blame)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
// Copyright 2019 yuzu emulator team
// Licensed under GPLv2 or any later version
// Refer to the license.txt file included.

#pragma once

#include <map>
#include <ostream>
#include <vector>
#include "common/bit_field.h"
#include "common/common_types.h"
#include "common/swap.h"

namespace Core {
class Reporter;
}

namespace Service::LM {

enum class DestinationFlag : u32 {
    Default = 1,
    UART = 2,
    UARTSleeping = 4,

    All = 0xFFFF,
};

struct MessageHeader {
    enum Flags : u32_le {
        IsHead = 1,
        IsTail = 2,
    };
    enum Severity : u32_le {
        Trace,
        Info,
        Warning,
        Error,
        Critical,
    };

    u64_le pid;
    u64_le thread_context;
    union {
        BitField<0, 16, Flags> flags;
        BitField<16, 8, Severity> severity;
        BitField<24, 8, u32> verbosity;
    };
    u32_le payload_size;

    bool IsHeadLog() const {
        return flags & IsHead;
    }
    bool IsTailLog() const {
        return flags & IsTail;
    }
};
static_assert(sizeof(MessageHeader) == 0x18, "MessageHeader is incorrect size");

enum class Field : u8 {
    Skip = 1,
    Message = 2,
    Line = 3,
    Filename = 4,
    Function = 5,
    Module = 6,
    Thread = 7,
};

std::ostream& operator<<(std::ostream& os, DestinationFlag dest);
std::ostream& operator<<(std::ostream& os, MessageHeader::Severity severity);
std::ostream& operator<<(std::ostream& os, Field field);

using FieldMap = std::map<Field, std::vector<u8>>;

struct LogMessage {
    MessageHeader header;
    FieldMap fields;
};

std::string FormatField(Field type, const std::vector<u8>& data);

class Manager {
public:
    explicit Manager(Core::Reporter& reporter);
    ~Manager();

    void SetEnabled(bool enabled);
    void SetDestination(DestinationFlag destination);

    void Log(LogMessage message);

    void Flush();

private:
    void InitializeLog();
    void FinalizeLog();

    bool enabled = true;
    DestinationFlag destination = DestinationFlag::All;

    std::vector<LogMessage> current_log;

    Core::Reporter& reporter;
};

} // namespace Service::LM