aboutsummaryrefslogtreecommitdiffstats
path: root/library/cpp/dbg_output/dump.h
blob: 448f1a8f5ae523d67ab3b03ebbda9e72c48caf6e (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
#pragma once

#include "engine.h"
#include "dumpers.h"
#include "auto.h"
#include "colorscheme.h"

#include <util/stream/format.h>
#include <util/system/type_name.h>
#include <util/generic/hash_set.h>
#include <utility>

/*
 * Cout << DbgDump(any) << Endl;
 * Cout << DbgDumpDeep(any) << Endl;
 * Cout << DbgDump(any).SetIndent(true) << Endl;
 *
 * specialize TDumper<your type> for extending dumper
 */

namespace NPrivate {
    template <class TColorScheme>
    struct TTraitsShallow {
        struct TDump: public TDumpBase, public TColorSchemeContainer<TColorScheme> {
            template <typename... Args>
            inline TDump(Args&&... args)
                : TDumpBase(std::forward<Args>(args)...)
            {
            }

            template <class V>
            inline void Pointer(const V* v) {
                if (v) {
                    *this << DumpRaw("(") << DumpRaw(TypeName(v).data()) << DumpRaw(")") << Hex((size_t)v);
                } else {
                    *this << DumpRaw("(") << DumpRaw(TypeName<V>().data()) << DumpRaw("*)nullptr");
                }
            }
        };
    };

    template <class TColorScheme>
    struct TTraitsDeep {
        struct TDump: public TDumpBase, public TColorSchemeContainer<TColorScheme> {
            template <typename... Args>
            inline TDump(Args&&... args)
                : TDumpBase(std::forward<Args>(args)...)
            {
            }

            template <class V>
            inline void Pointer(const V* v) {
                if (v && !Visited.contains((size_t)v)) {
                    Visited.insert((size_t)v);
                    *this << DumpRaw("(") << DumpRaw(TypeName(v).data()) << DumpRaw(")") << Hex((size_t)v) << DumpRaw(" -> ") << *v;
                    Visited.erase((size_t)v);
                } else {
                    *this << DumpRaw("(") << DumpRaw(TypeName<V>().data()) << DumpRaw("*)nullptr");
                }
            }

            THashSet<size_t> Visited;
        };
    };

    template <class T, class TTraits>
    struct TDbgDump {
        inline TDbgDump(const T* t)
            : T_(t)
            , Indent(false)
        {
        }

        inline void DumpTo(IOutputStream& out) const { 
            typename TTraits::TDump d(out, Indent);

            d << *T_;
        }

        inline TDbgDump& SetIndent(bool v) noexcept {
            Indent = v;

            return *this;
        }

        const T* T_;
        bool Indent;
    };

    template <class T, class TTraits>
    static inline IOutputStream& operator<<(IOutputStream& out, const TDbgDump<T, TTraits>& d) { 
        d.DumpTo(out);

        return out;
    }
}

template <class T, class TColorScheme = DBG_OUTPUT_DEFAULT_COLOR_SCHEME>
static inline ::NPrivate::TDbgDump<T, ::NPrivate::TTraitsShallow<TColorScheme>> DbgDump(const T& t) {
    return {std::addressof(t)};
}

template <class T, class TColorScheme = DBG_OUTPUT_DEFAULT_COLOR_SCHEME>
static inline ::NPrivate::TDbgDump<T, ::NPrivate::TTraitsDeep<TColorScheme>> DbgDumpDeep(const T& t) {
    return {std::addressof(t)};
}