• Home
  • Features
  • Pricing
  • Docs
  • Announcements
  • Sign In

daisytuner / docc / 27237522183

09 Jun 2026 09:38PM UTC coverage: 61.388% (+0.1%) from 61.275%
27237522183

Pull #741

github

web-flow
Merge c8be834b4 into aacd50c09
Pull Request #741: replaces MemAccessRangeAnalysis with MemoryLayoutAnalysis

481 of 523 new or added lines in 12 files covered. (91.97%)

41 existing lines in 10 files now uncovered.

35748 of 58233 relevant lines covered (61.39%)

757.2 hits per line

Source File
Press 'n' to go to next uncovered line, 'b' for previous

63.58
/sdfg/src/visualizer/visualizer.cpp
1
#include "sdfg/visualizer/visualizer.h"
2

3
#include <cassert>
4
#include <sstream>
5
#include <stdexcept>
6
#include <string>
7
#include <utility>
8
#include <vector>
9

10
#include "sdfg/data_flow/tasklet.h"
11
#include "sdfg/helpers/helpers.h"
12
#include "sdfg/structured_control_flow/block.h"
13
#include "sdfg/structured_control_flow/control_flow_node.h"
14
#include "sdfg/structured_control_flow/for.h"
15
#include "sdfg/structured_control_flow/if_else.h"
16
#include "sdfg/structured_control_flow/return.h"
17
#include "sdfg/structured_control_flow/sequence.h"
18
#include "sdfg/structured_control_flow/while.h"
19
#include "sdfg/structured_sdfg.h"
20
#include "sdfg/symbolic/symbolic.h"
21
#include "sdfg/types/type.h"
22
#include "symengine/basic.h"
23

24
namespace sdfg {
25
namespace visualizer {
26

27
constexpr const char* code_to_string(data_flow::TaskletCode c) {
17✔
28
    switch (c) {
17✔
29
        case data_flow::TaskletCode::assign:
7✔
30
            return "=";
7✔
31
        case data_flow::TaskletCode::int_add:
1✔
32
        case data_flow::TaskletCode::fp_add:
3✔
33
            return "+";
3✔
34
        case data_flow::TaskletCode::int_sub:
1✔
35
        case data_flow::TaskletCode::fp_sub:
2✔
36
            return "-";
2✔
37
        case data_flow::TaskletCode::int_mul:
×
38
        case data_flow::TaskletCode::fp_mul:
4✔
39
            return "*";
4✔
40
        case data_flow::TaskletCode::int_udiv:
×
41
        case data_flow::TaskletCode::int_sdiv:
×
42
        case data_flow::TaskletCode::fp_div:
×
43
            return "/";
×
44
        case data_flow::TaskletCode::int_urem:
×
45
        case data_flow::TaskletCode::int_srem:
×
46
        case data_flow::TaskletCode::fp_rem:
×
47
            return "%";
×
48
        case data_flow::TaskletCode::fp_fma:
1✔
49
            return "fma";
1✔
50
        case data_flow::TaskletCode::fp_one:
×
51
            return "of!=";
×
52
        default:
×
53
            return "?";
×
54
    };
17✔
55
};
×
56

57
std::string Visualizer::expression(const std::string expr) {
78✔
58
    if (this->replacements_.empty()) return expr;
78✔
59
    std::string res = expr;
×
60
    size_t pos1 = 0, pos2 = 0;
×
61
    for (std::pair<const std::string, const std::string> replace : this->replacements_) {
×
62
        pos2 = res.find(replace.first);
×
63
        if (pos2 == res.npos) continue;
×
64
        pos1 = 0;
×
65
        std::stringstream res_tmp;
×
66
        while (pos2 < res.npos) {
×
67
            res_tmp << res.substr(pos1, pos2 - pos1) << replace.second;
×
68
            pos1 = pos2 + replace.first.size();
×
69
            pos2 = res.find(replace.first, pos1);
×
70
        }
×
71
        if (pos1 < res.npos) res_tmp << res.substr(pos1);
×
72
        res = res_tmp.str();
×
73
    }
×
74
    return res;
×
75
}
78✔
76

77
void Visualizer::visualizeNode(const StructuredSDFG& sdfg, const structured_control_flow::ControlFlowNode& node) {
33✔
78
    if (auto block = dynamic_cast<const structured_control_flow::Block*>(&node)) {
33✔
79
        this->visualizeBlock(sdfg, *block);
18✔
80
        return;
18✔
81
    }
18✔
82
    if (auto sequence = dynamic_cast<const structured_control_flow::Sequence*>(&node)) {
15✔
83
        this->visualizeSequence(sdfg, *sequence);
×
84
        return;
×
85
    }
×
86
    if (auto if_else = dynamic_cast<const structured_control_flow::IfElse*>(&node)) {
15✔
87
        this->visualizeIfElse(sdfg, *if_else);
3✔
88
        return;
3✔
89
    }
3✔
90
    if (auto while_loop = dynamic_cast<const structured_control_flow::While*>(&node)) {
12✔
91
        this->visualizeWhile(sdfg, *while_loop);
2✔
92
        return;
2✔
93
    }
2✔
94
    if (auto loop = dynamic_cast<const structured_control_flow::For*>(&node)) {
10✔
95
        this->visualizeFor(sdfg, *loop);
6✔
96
        return;
6✔
97
    }
6✔
98
    if (auto return_node = dynamic_cast<const structured_control_flow::Return*>(&node)) {
4✔
99
        this->visualizeReturn(sdfg, *return_node);
2✔
100
        return;
2✔
101
    }
2✔
102
    if (auto break_node = dynamic_cast<const structured_control_flow::Break*>(&node)) {
2✔
103
        this->visualizeBreak(sdfg, *break_node);
1✔
104
        return;
1✔
105
    }
1✔
106
    if (auto continue_node = dynamic_cast<const structured_control_flow::Continue*>(&node)) {
1✔
107
        this->visualizeContinue(sdfg, *continue_node);
1✔
108
        return;
1✔
109
    }
1✔
UNCOV
110
    if (auto map_node = dynamic_cast<const structured_control_flow::Map*>(&node)) {
×
UNCOV
111
        this->visualizeMap(sdfg, *map_node);
×
UNCOV
112
        return;
×
UNCOV
113
    }
×
114
    throw std::runtime_error("Unsupported control flow node");
×
UNCOV
115
}
×
116

117
void Visualizer::visualizeTasklet(data_flow::Tasklet const& tasklet) {
17✔
118
    std::string op = code_to_string(tasklet.code());
17✔
119
    std::vector<std::string> arguments;
17✔
120
    for (size_t i = 0; i < tasklet.inputs().size(); ++i) {
45✔
121
        arguments.push_back(this->expression(tasklet.input(i)));
28✔
122
    }
28✔
123

124
    if (tasklet.code() == data_flow::TaskletCode::assign) {
17✔
125
        this->stream_ << arguments.at(0);
7✔
126
    } else if (tasklet.code() == data_flow::TaskletCode::fp_fma) {
10✔
127
        if (arguments.size() != 3) throw std::runtime_error("FMA requires 3 arguments");
1✔
128
        this->stream_ << arguments.at(0) << " * " << arguments.at(1) << " + " << arguments.at(2);
1✔
129
    } else {
9✔
130
        this->stream_ << op << "(" << helpers::join(arguments, ", ") << ")";
9✔
131
    }
9✔
132
}
17✔
133

134
void Visualizer::visualizeForBounds(
135
    symbolic::Symbol const& indvar,
136
    symbolic::Expression const& init,
137
    symbolic::Condition const& condition,
138
    symbolic::Expression const& update
139
) {
6✔
140
    this->stream_ << indvar->get_name() << " = " << this->expression(init->__str__()) << "; "
6✔
141
                  << this->expression(condition->__str__()) << "; " << indvar->get_name() << " = "
6✔
142
                  << this->expression(update->__str__());
6✔
143
}
6✔
144

145
std::string Visualizer::subsetRangeString(data_flow::Subset const& subset, int subIdx) {
26✔
146
    auto& dim = subset.at(subIdx);
26✔
147
    return this->expression(dim->__str__());
26✔
148
}
26✔
149

150
/// @brief If known, use the type to better visualize structures. Then track the type as far as it goes.
151
void Visualizer::visualizeSubset(data_flow::Subset const& sub, types::IType const* type, int subIdx) {
65✔
152
    if (static_cast<int>(sub.size()) <= subIdx) {
65✔
153
        return;
39✔
154
    }
39✔
155
    if (auto structure_type = dynamic_cast<const types::Structure*>(type)) {
26✔
156
        types::StructureDefinition const& definition = this->sdfg_.structure(structure_type->name());
×
157

158
        auto& memberIdx = sub.at(subIdx);
×
159
        if (!memberIdx.is_null() && SymEngine::is_a<SymEngine::Integer>(*memberIdx)) {
×
160
            this->stream_ << ".member_" << this->expression(memberIdx->__str__());
×
161
            auto& member_type = definition.member_type(SymEngine::rcp_dynamic_cast<const SymEngine::Integer>(memberIdx)
×
162
            );
×
163
            this->visualizeSubset(sub, &member_type, subIdx + 1);
×
164
        } else {
×
165
            this->stream_ << ".member[" << subsetRangeString(sub, subIdx) << "]";
×
166
            this->visualizeSubset(sub, nullptr, subIdx + 1);
×
167
        }
×
168
    } else if (auto tensor_type = dynamic_cast<const types::Tensor*>(type)) {
26✔
169
        auto& shape = tensor_type->shape();
×
170
        int tensor_dims = shape.size();
×
171
        int i = 0;
×
172
        while (i < tensor_dims && subIdx < sub.size()) {
×
173
            this->stream_ << "[" << subsetRangeString(sub, i) << ":" << subsetRangeString(shape, i) << "]";
×
174
            ++subIdx;
×
175
            ++i;
×
176
        }
×
177
        if (subIdx < sub.size()) {
×
178
            this->visualizeSubset(sub, &tensor_type->element_type(), subIdx);
×
179
        }
×
180
    } else if (auto array_type = dynamic_cast<const types::Array*>(type)) {
26✔
181
        this->stream_ << "[" << subsetRangeString(sub, subIdx) << "]";
14✔
182
        types::IType const& element_type = array_type->element_type();
14✔
183
        this->visualizeSubset(sub, &element_type, subIdx + 1);
14✔
184
    } else if (auto pointer_type = dynamic_cast<const types::Pointer*>(type)) {
14✔
185
        this->stream_ << "[" << subsetRangeString(sub, subIdx) << "]";
11✔
186
        const types::IType* pointee_type;
11✔
187
        if (pointer_type->has_pointee_type()) {
11✔
188
            pointee_type = &pointer_type->pointee_type();
8✔
189
        } else {
8✔
190
            auto z = symbolic::zero();
3✔
191
            if (!symbolic::eq(sub.at(subIdx), z)) {
3✔
192
                this->stream_ << "#illgl";
1✔
193
            }
1✔
194
            pointee_type = nullptr;
3✔
195
        }
3✔
196
        this->visualizeSubset(sub, pointee_type, subIdx + 1);
11✔
197
    } else {
11✔
198
        if (type == nullptr) {
1✔
199
            this->stream_ << "(rogue)";
1✔
200
        }
1✔
201
        this->stream_ << "[" << subsetRangeString(sub, subIdx) << "]";
1✔
202
        visualizeSubset(sub, nullptr, subIdx + 1);
1✔
203
    }
1✔
204
}
26✔
205

206
void Visualizer::visualize() {
13✔
207
    if (const auto* unstructured_sdfg = dynamic_cast<const SDFG*>(&this->sdfg_)) {
13✔
208
        this->visualizeSDFG(*unstructured_sdfg);
1✔
209
    } else if (const auto* structured_sdfg = dynamic_cast<const StructuredSDFG*>(&this->sdfg_)) {
12✔
210
        this->visualizeStructuredSDFG(*structured_sdfg);
12✔
211
    } else {
12✔
212
        throw InvalidSDFGException("Visualizer: Invalid SDFG type");
×
213
    }
×
214
}
13✔
215

216
} // namespace visualizer
217
} // namespace sdfg
STATUS · Troubleshooting · Open an Issue · Sales · Support · CAREERS · ENTERPRISE · START FREE · SCHEDULE DEMO
ANNOUNCEMENTS · TWITTER · TOS & SLA · Supported CI Services · What's a CI service? · Automated Testing

© 2026 Coveralls, Inc