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

daisytuner / sdfglib / 20283073339

16 Dec 2025 09:20PM UTC coverage: 40.075% (-0.01%) from 40.086%
20283073339

push

github

web-flow
Merge pull request #395 from daisytuner/perf-improvs

Various performance improvements

13514 of 43711 branches covered (30.92%)

Branch coverage included in aggregate %.

132 of 169 new or added lines in 9 files covered. (78.11%)

11 existing lines in 5 files now uncovered.

11632 of 19037 relevant lines covered (61.1%)

87.78 hits per line

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

54.41
/src/analysis/users.cpp
1
#include "sdfg/analysis/users.h"
2

3
#include <cassert>
4
#include <list>
5
#include <memory>
6
#include <string>
7
#include <unordered_map>
8
#include <unordered_set>
9
#include <vector>
10

11
#include "sdfg/data_flow/memlet.h"
12
#include "sdfg/element.h"
13
#include "sdfg/graph/graph.h"
14
#include "sdfg/structured_control_flow/for.h"
15
#include "sdfg/structured_control_flow/map.h"
16
#include "sdfg/structured_control_flow/sequence.h"
17
#include "sdfg/structured_sdfg.h"
18
#include "sdfg/symbolic/sets.h"
19
#include "sdfg/symbolic/symbolic.h"
20

21
namespace sdfg {
22
namespace analysis {
23

24
User::User(graph::Vertex vertex, const std::string& container, Element* element, Use use)
6,662✔
25
    : vertex_(vertex), container_(container), use_(use), element_(element) {
3,331✔
26

27
      };
28

29
Use User::use() const { return this->use_; };
11,135✔
30

31
std::string& User::container() { return this->container_; };
29,234✔
32

33
Element* User::element() { return this->element_; };
3,011✔
34

35
const std::vector<data_flow::Subset>& User::subsets() const {
321✔
36
    if (this->subsets_cached_) {
321✔
37
        return this->subsets_;
178✔
38
    }
39

40
    if (this->container_ == "") {
143!
41
        // No-op user
42
    } else if (auto access_node = dynamic_cast<data_flow::AccessNode*>(this->element_)) {
143!
43
        auto& graph = access_node->get_parent();
115✔
44
        if (this->use_ == Use::READ || this->use_ == Use::VIEW) {
115!
45
            for (auto& iedge : graph.out_edges(*access_node)) {
73✔
46
                this->subsets_.push_back(iedge.subset());
37✔
47
            }
48
        } else if (this->use_ == Use::WRITE || this->use_ == Use::MOVE) {
115!
49
            for (auto& oedge : graph.in_edges(*access_node)) {
158✔
50
                this->subsets_.push_back(oedge.subset());
79✔
51
            }
52
        }
79✔
53
    } else {
115✔
54
        // Use of symbol
55
        this->subsets_.push_back({});
28!
56
    }
57

58
    this->subsets_cached_ = true;
143✔
59
    return this->subsets_;
143✔
60
};
321✔
61

62
ForUser::ForUser(
647✔
63
    graph::Vertex vertex,
64
    const std::string& container,
65
    Element* element,
66
    Use use,
67
    bool is_init,
68
    bool is_condition,
69
    bool is_update
70
)
71
    : User(vertex, container, element, use), is_init_(is_init), is_condition_(is_condition), is_update_(is_update) {
647✔
72

73
      };
647✔
74

75
bool ForUser::is_init() const { return this->is_init_; };
651✔
76

77
bool ForUser::is_condition() const { return this->is_condition_; };
507✔
78

79
bool ForUser::is_update() const { return this->is_update_; };
263✔
80

81
std::pair<graph::Vertex, graph::Vertex> Users::traverse(data_flow::DataFlowGraph& dataflow) {
325✔
82
    graph::Vertex first = boost::graph_traits<graph::Graph>::null_vertex();
325✔
83
    graph::Vertex last = boost::graph_traits<graph::Graph>::null_vertex();
325✔
84
    for (auto node : dataflow.topological_sort()) {
1,002✔
85
        if (dynamic_cast<data_flow::ConstantNode*>(node) != nullptr) {
677!
86
            continue;
33✔
87
        }
88

89
        if (auto access_node = dynamic_cast<data_flow::AccessNode*>(node)) {
644!
90
            if (!symbolic::is_pointer(symbolic::symbol(access_node->data()))) {
446!
91
                if (dataflow.in_degree(*node) > 0) {
446!
92
                    Use use = Use::WRITE;
236✔
93

94
                    // Check if the pointer itself is moved (overwritten)
95
                    for (auto& iedge : dataflow.in_edges(*access_node)) {
438!
96
                        if (iedge.type() == data_flow::MemletType::Reference ||
445!
97
                            iedge.type() == data_flow::MemletType::Dereference_Src) {
209!
98
                            use = Use::MOVE;
34✔
99
                            break;
34✔
100
                        }
101
                    }
102

103
                    auto v = boost::add_vertex(this->graph_);
236!
104
                    this->add_user(std::make_unique<User>(v, access_node->data(), access_node, use));
236!
105

106
                    if (last != boost::graph_traits<graph::Graph>::null_vertex()) {
236!
107
                        boost::add_edge(last, v, this->graph_);
204!
108
                    } else {
204✔
109
                        first = v;
32✔
110
                    }
111
                    last = v;
236✔
112
                }
236✔
113
                if (dataflow.out_degree(*access_node) > 0) {
446!
114
                    Use use = Use::READ;
225✔
115

116
                    // Check if the pointer itself is viewed (aliased)
117
                    for (auto& oedge : dataflow.out_edges(*access_node)) {
421!
118
                        if (oedge.type() == data_flow::MemletType::Reference ||
429!
119
                            oedge.type() == data_flow::MemletType::Dereference_Dst) {
201!
120
                            use = Use::VIEW;
32✔
121
                            break;
32✔
122
                        }
123
                    }
124

125
                    auto v = boost::add_vertex(this->graph_);
225!
126
                    this->add_user(std::make_unique<User>(v, access_node->data(), access_node, use));
225!
127

128
                    if (last != boost::graph_traits<graph::Graph>::null_vertex()) {
225!
129
                        boost::add_edge(last, v, this->graph_);
52!
130
                    } else {
52✔
131
                        first = v;
173✔
132
                    }
133
                    last = v;
225✔
134
                }
225✔
135
            }
446✔
136
        } else if (auto library_node = dynamic_cast<data_flow::LibraryNode*>(node)) {
644!
137
            for (auto& symbol : library_node->symbols()) {
12!
138
                auto v = boost::add_vertex(this->graph_);
×
139
                this->add_user(std::make_unique<User>(v, symbol->get_name(), library_node, Use::READ));
×
140
                if (last != boost::graph_traits<graph::Graph>::null_vertex()) {
×
141
                    boost::add_edge(last, v, this->graph_);
×
142
                } else {
×
143
                    first = v;
×
144
                }
145
                last = v;
×
146
            }
147
        }
12✔
148

149
        for (auto& oedge : dataflow.out_edges(*node)) {
1,070!
150
            std::unordered_set<std::string> used;
426✔
151
            for (auto dim : oedge.subset()) {
762!
152
                for (auto atom : symbolic::atoms(dim)) {
546!
153
                    if (used.find(atom->get_name()) != used.end()) {
210!
154
                        continue;
×
155
                    }
156
                    used.insert(atom->get_name());
210!
157

158
                    auto v = boost::add_vertex(this->graph_);
210!
159
                    this->add_user(std::make_unique<User>(v, atom->get_name(), &oedge, Use::READ));
210!
160
                    if (last != boost::graph_traits<graph::Graph>::null_vertex()) {
210!
161
                        boost::add_edge(last, v, this->graph_);
197!
162
                    } else {
197✔
163
                        first = v;
13✔
164
                    }
165
                    last = v;
210✔
166
                }
210!
167
            }
336✔
168
        }
426✔
169
    }
170

171
    return {first, last};
325✔
172
};
×
173

174
std::pair<graph::Vertex, graph::Vertex> Users::traverse(structured_control_flow::ControlFlowNode& node) {
926✔
175
    if (auto block_stmt = dynamic_cast<structured_control_flow::Block*>(&node)) {
926!
176
        // NOP
177
        auto s = boost::add_vertex(this->graph_);
325✔
178
        this->users_.insert({s, std::make_unique<User>(s, "", block_stmt, Use::NOP)});
325!
179
        this->entries_.insert({block_stmt, this->users_.at(s).get()});
325✔
180

181
        // NOP
182
        auto t = boost::add_vertex(this->graph_);
325✔
183
        this->users_.insert({t, std::make_unique<User>(t, "", block_stmt, Use::NOP)});
325!
184
        this->exits_.insert({block_stmt, this->users_.at(t).get()});
325✔
185

186
        auto& dataflow = block_stmt->dataflow();
325✔
187
        auto subgraph = this->traverse(dataflow);
325✔
188

189
        // May be empty
190
        if (subgraph.first == boost::graph_traits<graph::Graph>::null_vertex()) {
325✔
191
            boost::add_edge(s, t, this->graph_);
107✔
192
            return {s, t};
107✔
193
        }
194

195
        boost::add_edge(s, subgraph.first, this->graph_);
218✔
196
        boost::add_edge(subgraph.second, t, this->graph_);
218✔
197

198
        return {s, t};
218✔
199
    } else if (auto sequence_stmt = dynamic_cast<structured_control_flow::Sequence*>(&node)) {
601!
200
        auto s = boost::add_vertex(this->graph_);
424✔
201
        this->users_.insert({s, std::make_unique<User>(s, "", sequence_stmt, Use::NOP)});
424!
202
        this->entries_.insert({sequence_stmt, this->users_.at(s).get()});
424✔
203

204
        graph::Vertex current = s;
424✔
205
        for (size_t i = 0; i < sequence_stmt->size(); i++) {
930✔
206
            auto child = sequence_stmt->at(i);
508✔
207

208
            auto subgraph = this->traverse(child.first);
508✔
209
            boost::add_edge(current, subgraph.first, this->graph_);
508✔
210
            // Return node
211
            if (subgraph.second == boost::graph_traits<graph::Graph>::null_vertex()) {
508✔
212
                break;
2✔
213
            }
214
            current = subgraph.second;
506✔
215

216
            std::unordered_set<std::string> used;
506✔
217
            for (auto& entry : child.second.assignments()) {
612!
218
                for (auto atom : symbolic::atoms(entry.second)) {
145!
219
                    if (symbolic::is_pointer(atom)) {
39!
220
                        continue;
×
221
                    }
222
                    if (used.find(atom->get_name()) != used.end()) {
39!
223
                        continue;
×
224
                    }
225
                    used.insert(atom->get_name());
39!
226

227
                    auto v = boost::add_vertex(this->graph_);
39!
228
                    this->add_user(std::make_unique<User>(v, atom->get_name(), &child.second, Use::READ));
39!
229

230
                    boost::add_edge(current, v, this->graph_);
39!
231
                    current = v;
39✔
232
                }
39!
233
            }
234

235
            for (auto& entry : child.second.assignments()) {
612!
236
                auto v = boost::add_vertex(this->graph_);
106!
237
                this->add_user(std::make_unique<User>(v, entry.first->get_name(), &child.second, Use::WRITE));
106!
238

239
                boost::add_edge(current, v, this->graph_);
106!
240
                current = v;
106✔
241
            }
242
        }
506✔
243

244
        if (current == boost::graph_traits<graph::Graph>::null_vertex()) {
424!
245
            this->exits_.insert({sequence_stmt, this->users_.at(s).get()});
×
246
            return {s, current};
×
247
        }
248

249
        auto t = boost::add_vertex(this->graph_);
424✔
250
        this->users_.insert({t, std::make_unique<User>(t, "", sequence_stmt, Use::NOP)});
424!
251
        boost::add_edge(current, t, this->graph_);
424✔
252
        this->exits_.insert({sequence_stmt, this->users_.at(t).get()});
424✔
253

254
        return {s, t};
424✔
255
    } else if (auto if_else_stmt = dynamic_cast<structured_control_flow::IfElse*>(&node)) {
177!
256
        // NOP
257
        auto s = boost::add_vertex(this->graph_);
28✔
258
        this->users_.insert({s, std::make_unique<User>(s, "", if_else_stmt, Use::NOP)});
28!
259
        this->entries_.insert({if_else_stmt, this->users_.at(s).get()});
28✔
260

261
        graph::Vertex last = s;
28✔
262

263
        std::unordered_set<std::string> used;
28✔
264
        for (size_t i = 0; i < if_else_stmt->size(); i++) {
73!
265
            auto condition = if_else_stmt->at(i).second;
45!
266
            for (auto atom : symbolic::atoms(condition)) {
77!
267
                if (used.find(atom->get_name()) != used.end()) {
32!
268
                    continue;
11✔
269
                }
270
                used.insert(atom->get_name());
21!
271

272
                auto v = boost::add_vertex(this->graph_);
21!
273

274
                this->add_user(std::make_unique<User>(v, atom->get_name(), if_else_stmt, Use::READ));
21!
275

276
                boost::add_edge(last, v, this->graph_);
21!
277
                last = v;
21✔
278
            }
32✔
279
        }
45✔
280

281
        graph::Vertex t = boost::graph_traits<graph::Graph>::null_vertex();
28!
282
        for (size_t i = 0; i < if_else_stmt->size(); i++) {
73!
283
            auto branch = if_else_stmt->at(i);
45!
284
            auto subgraph = this->traverse(branch.first);
45!
285
            boost::add_edge(last, subgraph.first, this->graph_);
45!
286
            if (subgraph.second != boost::graph_traits<graph::Graph>::null_vertex()) {
45!
287
                if (t == boost::graph_traits<graph::Graph>::null_vertex()) {
45!
288
                    t = boost::add_vertex(this->graph_);
28!
289
                    this->users_.insert({t, std::make_unique<User>(t, "", if_else_stmt, Use::NOP)});
28!
290
                    this->exits_.insert({if_else_stmt, this->users_.at(t).get()});
28!
291
                }
28✔
292
                boost::add_edge(subgraph.second, t, this->graph_);
45!
293
            }
45✔
294
        }
45✔
295

296
        // Forward edge: Potentially missing else case
297
        if (!if_else_stmt->is_complete()) {
28!
298
            if (t == boost::graph_traits<graph::Graph>::null_vertex()) {
11!
299
                t = boost::add_vertex(this->graph_);
×
300
                this->users_.insert({t, std::make_unique<User>(t, "", if_else_stmt, Use::NOP)});
×
301
                this->exits_.insert({if_else_stmt, this->users_.at(t).get()});
×
302
            }
×
303
            boost::add_edge(last, t, this->graph_);
11!
304
        }
11✔
305

306
        return {s, t};
28✔
307
    } else if (auto loop_stmt = dynamic_cast<structured_control_flow::While*>(&node)) {
177!
308
        // NOP
309
        auto s = boost::add_vertex(this->graph_);
9✔
310
        this->users_.insert({s, std::make_unique<User>(s, "", loop_stmt, Use::NOP)});
9!
311
        this->entries_.insert({loop_stmt, this->users_.at(s).get()});
9✔
312

313
        auto subgraph = this->traverse(loop_stmt->root());
9✔
314

315
        // NOP
316
        auto t = boost::add_vertex(this->graph_);
9✔
317
        this->users_.insert({t, std::make_unique<User>(t, "", loop_stmt, Use::NOP)});
9!
318
        this->exits_.insert({loop_stmt, this->users_.at(t).get()});
9✔
319

320
        boost::add_edge(s, subgraph.first, this->graph_);
9✔
321
        if (subgraph.second != boost::graph_traits<graph::Graph>::null_vertex()) {
9!
322
            boost::add_edge(subgraph.second, t, this->graph_);
9✔
323
        }
9✔
324

325
        // Empty loop
326
        boost::add_edge(s, t, this->graph_);
9✔
327
        // Back edge
328
        boost::add_edge(t, s, this->graph_);
9✔
329

330
        return {s, t};
9✔
331
    } else if (auto for_stmt = dynamic_cast<structured_control_flow::StructuredLoop*>(&node)) {
140!
332
        // NOP
333
        auto s = boost::add_vertex(this->graph_);
129✔
334
        this->users_.insert({s, std::make_unique<User>(s, "", for_stmt, Use::NOP)});
129!
335
        auto last = s;
129✔
336
        this->entries_.insert({for_stmt, this->users_.at(s).get()});
129✔
337

338
        // NOP
339
        auto t = boost::add_vertex(this->graph_);
129✔
340
        this->users_.insert({t, std::make_unique<User>(t, "", for_stmt, Use::NOP)});
129!
341
        this->exits_.insert({for_stmt, this->users_.at(t).get()});
129✔
342

343
        // Init
344
        for (auto atom : symbolic::atoms(for_stmt->init())) {
143!
345
            auto v = boost::add_vertex(this->graph_);
14!
346
            this->add_user(std::make_unique<ForUser>(v, atom->get_name(), for_stmt, Use::READ, true, false, false));
14!
347
            boost::add_edge(last, v, this->graph_);
14!
348
            last = v;
14✔
349
        }
14✔
350
        // Indvar
351
        auto v = boost::add_vertex(this->graph_);
129✔
352
        this->add_user(std::make_unique<
129!
353
                       ForUser>(v, for_stmt->indvar()->get_name(), for_stmt, Use::WRITE, true, false, false));
129!
354

355
        boost::add_edge(last, v, this->graph_);
129✔
356
        last = v;
129✔
357

358
        // Condition
359
        for (auto atom : symbolic::atoms(for_stmt->condition())) {
375!
360
            auto v = boost::add_vertex(this->graph_);
246!
361
            this->add_user(std::make_unique<ForUser>(v, atom->get_name(), for_stmt, Use::READ, false, true, false));
246!
362

363
            boost::add_edge(last, v, this->graph_);
246!
364
            boost::add_edge(v, t, this->graph_);
246!
365
            last = v;
246✔
366
        }
246✔
367

368
        auto subgraph = this->traverse(for_stmt->root());
129✔
369
        boost::add_edge(last, subgraph.first, this->graph_);
129✔
370

371
        // Update
372
        auto end = subgraph.second;
129✔
373
        for (auto atom : symbolic::atoms(for_stmt->update())) {
258!
374
            auto v = boost::add_vertex(this->graph_);
129!
375
            this->add_user(std::make_unique<ForUser>(v, atom->get_name(), for_stmt, Use::READ, false, false, true));
129!
376
            boost::add_edge(end, v, this->graph_);
129!
377
            end = v;
129✔
378
        }
129✔
379

380
        auto update_v = boost::add_vertex(this->graph_);
129✔
381
        this->add_user(std::make_unique<
129!
382
                       ForUser>(update_v, for_stmt->indvar()->get_name(), for_stmt, Use::WRITE, false, false, true));
129!
383

384
        if (end != boost::graph_traits<graph::Graph>::null_vertex()) {
129!
385
            boost::add_edge(end, update_v, this->graph_);
129✔
386
        }
129✔
387
        end = update_v;
129✔
388

389
        if (end != boost::graph_traits<graph::Graph>::null_vertex()) {
129!
390
            boost::add_edge(end, t, this->graph_);
129✔
391
        }
129✔
392

393
        // Back edge
394
        boost::add_edge(t, last, this->graph_);
129✔
395

396
        return {s, t};
129✔
397
    } else if (auto cont_stmt = dynamic_cast<structured_control_flow::Continue*>(&node)) {
11!
398
        // Approximated by general back edge in loop scope
399
        auto v = boost::add_vertex(this->graph_);
4✔
400
        this->users_.insert({v, std::make_unique<User>(v, "", cont_stmt, Use::NOP)});
4!
401
        this->entries_.insert({cont_stmt, this->users_.at(v).get()});
4✔
402
        this->exits_.insert({cont_stmt, this->users_.at(v).get()});
4✔
403
        return {v, v};
4✔
404
    } else if (auto br_stmt = dynamic_cast<structured_control_flow::Break*>(&node)) {
7!
405
        // Approximated by general back edge in loop scope
406
        auto v = boost::add_vertex(this->graph_);
5✔
407
        this->users_.insert({v, std::make_unique<User>(v, "", br_stmt, Use::NOP)});
5!
408
        this->entries_.insert({br_stmt, this->users_.at(v).get()});
5✔
409
        this->exits_.insert({br_stmt, this->users_.at(v).get()});
5✔
410
        return {v, v};
5✔
411
    } else if (auto ret_stmt = dynamic_cast<structured_control_flow::Return*>(&node)) {
2!
412
        if (!ret_stmt->is_data() || ret_stmt->data().empty()) {
2!
413
            auto v = boost::add_vertex(this->graph_);
×
414
            this->users_.insert({v, std::make_unique<User>(v, "", ret_stmt, Use::NOP)});
×
415
            this->entries_.insert({ret_stmt, this->users_.at(v).get()});
×
416
            this->exits_.insert({ret_stmt, this->users_.at(v).get()});
×
417
            return {v, boost::graph_traits<graph::Graph>::null_vertex()};
×
418
        } else {
419
            auto v = boost::add_vertex(this->graph_);
2✔
420
            this->add_user(std::make_unique<User>(v, ret_stmt->data(), ret_stmt, Use::READ));
2!
421
            this->entries_.insert({ret_stmt, this->users_.at(v).get()});
2✔
422
            this->exits_.insert({ret_stmt, this->users_.at(v).get()});
2✔
423
            return {v, boost::graph_traits<graph::Graph>::null_vertex()};
2✔
424
        }
425
    }
426

427
    throw std::invalid_argument("Invalid control flow node type");
×
428
};
926✔
429

430
Users::Users(StructuredSDFG& sdfg)
235!
431
    : Analysis(sdfg), node_(sdfg.root()) {
235!
432

433
      };
235✔
434

435
Users::Users(StructuredSDFG& sdfg, structured_control_flow::ControlFlowNode& node)
×
436
    : Analysis(sdfg), node_(node) {
×
437

438
      };
×
439

440
void Users::run(analysis::AnalysisManager& analysis_manager) {
235✔
441
    users_.clear();
235✔
442
    graph_.clear();
235✔
443
    source_ = nullptr;
235✔
444
    sink_ = nullptr;
235✔
445
    this->entries_.clear();
235✔
446
    this->exits_.clear();
235✔
447
    this->users_lookup_.clear();
235✔
448

449
    reads_.clear();
235✔
450
    writes_.clear();
235✔
451
    views_.clear();
235✔
452
    moves_.clear();
235✔
453

454
    this->traverse(node_);
235✔
455
    if (this->users_.empty()) {
235!
456
        return;
×
457
    }
458

459
    // Require a single source
460
    for (auto& entry : this->users_) {
3,560✔
461
        if (boost::in_degree(entry.first, this->graph_) == 0) {
3,325✔
462
            if (this->source_ != nullptr) {
235!
463
                throw InvalidSDFGException("Users Analysis: Non-unique source node");
×
464
            }
465
            this->source_ = entry.second.get();
235✔
466
        }
235✔
467
    }
468
    if (this->source_ == nullptr) {
235!
469
        throw InvalidSDFGException("Users Analysis: No source node");
×
470
    }
471

472
    std::list<User*> sinks;
235✔
473
    for (auto& entry : this->users_) {
3,560✔
474
        if (boost::out_degree(entry.first, this->graph_) == 0) {
3,325!
475
            sinks.push_back(entry.second.get());
237!
476
        }
237✔
477
    }
478
    if (sinks.size() == 0) {
235!
479
        throw InvalidSDFGException("Users Analysis: No sink node");
×
480
    }
481
    if (sinks.size() > 1) {
235✔
482
        // add artificial sink
483
        auto v = boost::add_vertex(this->graph_);
1!
484
        auto sink = std::make_unique<User>(v, "", nullptr, Use::NOP);
1!
485
        this->users_.insert({v, std::move(sink)});
1!
486
        for (auto sink : sinks) {
4✔
487
            boost::add_edge(sink->vertex_, v, this->graph_);
3!
488
        }
489
        sinks.clear();
1✔
490

491
        this->sink_ = this->users_.at(v).get();
1!
492
    } else {
1✔
493
        this->sink_ = sinks.front();
234✔
494
    }
495

496
    for (auto& container : sdfg_.containers()) {
904!
497
        this->reads_.insert({container, {}});
669!
498
        this->writes_.insert({container, {}});
669!
499
        this->views_.insert({container, {}});
669!
500
        this->moves_.insert({container, {}});
669!
501
    }
502

503
    // Collect sub structures
504
    for (auto& entry : this->users_) {
3,561✔
505
        auto container = entry.second->container();
3,326!
506
        if (entry.second->use() == Use::NOP) {
3,326✔
507
            continue;
1,840✔
508
        }
509
        if (container == "") {
1,486!
510
            continue;
×
511
        }
512

513
        switch (entry.second->use()) {
1,486!
514
            case Use::READ: {
515
                this->reads_[container].push_back(entry.second.get());
854!
516
                break;
854✔
517
            }
518
            case Use::WRITE: {
519
                this->writes_[container].push_back(entry.second.get());
566!
520
                break;
566✔
521
            }
522
            case Use::VIEW: {
523
                this->views_[container].push_back(entry.second.get());
32!
524
                break;
32✔
525
            }
526
            case Use::MOVE: {
527
                this->moves_[container].push_back(entry.second.get());
34!
528
                break;
34✔
529
            }
530
            default:
531
                break;
×
532
        }
533
    }
3,326✔
534
};
235✔
535

536
std::list<User*> Users::uses() const {
×
537
    std::list<User*> us;
×
538
    for (auto& entry : this->users_) {
×
539
        if (entry.second->use() == Use::NOP) {
×
540
            continue;
×
541
        }
542
        us.push_back(entry.second.get());
×
543
    }
544

545
    return us;
×
546
};
×
547

548
std::list<User*> Users::uses(const std::string& container) const {
262✔
549
    std::list<User*> us;
262✔
550
    for (auto& entry : this->users_) {
5,579✔
551
        if (entry.second->container() != container) {
5,317✔
552
            continue;
4,860✔
553
        }
554
        if (entry.second->use() == Use::NOP) {
457!
555
            continue;
×
556
        }
557
        us.push_back(entry.second.get());
457!
558
    }
559

560
    return us;
262✔
561
};
262!
562

563
size_t Users::num_uses(const std::string& container) const { return this->uses(container).size(); };
×
564

565
std::list<User*> Users::writes() const {
×
566
    std::list<User*> us;
×
567
    for (auto& entry : this->users_) {
×
568
        if (entry.second->use() != Use::WRITE) {
×
569
            continue;
×
570
        }
571
        us.push_back(entry.second.get());
×
572
    }
573

574
    return us;
×
575
};
×
576

577
const std::list<User*>& Users::writes(const std::string& container) const { return this->writes_.at(container); };
78✔
578

579
size_t Users::num_writes(const std::string& container) const { return this->writes(container).size(); };
28✔
580

581
std::list<User*> Users::reads() const {
×
582
    std::list<User*> us;
×
583
    for (auto& entry : this->users_) {
×
584
        if (entry.second->use() != Use::READ) {
×
585
            continue;
×
586
        }
587
        us.push_back(entry.second.get());
×
588
    }
589

590
    return us;
×
591
};
×
592

593
const std::list<User*>& Users::reads(const std::string& container) const { return this->reads_.at(container); };
156✔
594

595
size_t Users::num_reads(const std::string& container) const { return this->reads(container).size(); };
21✔
596

597
std::list<User*> Users::views() const {
×
598
    std::list<User*> us;
×
599
    for (auto& entry : this->users_) {
×
600
        if (entry.second->use() != Use::VIEW) {
×
601
            continue;
×
602
        }
603
        us.push_back(entry.second.get());
×
604
    }
605

606
    return us;
×
607
};
×
608

609
const std::list<User*>& Users::views(const std::string& container) const { return this->views_.at(container); };
58✔
610

611
size_t Users::num_views(const std::string& container) const { return this->views(container).size(); };
52✔
612

613
std::list<User*> Users::moves() const {
×
614
    std::list<User*> us;
×
615
    for (auto& entry : this->users_) {
×
616
        if (entry.second->use() != Use::MOVE) {
×
617
            continue;
×
618
        }
619
        us.push_back(entry.second.get());
×
620
    }
621

622
    return us;
×
623
};
×
624

625
const std::list<User*>& Users::moves(const std::string& container) const { return this->moves_.at(container); };
106✔
626

627
size_t Users::num_moves(const std::string& container) const { return this->moves(container).size(); };
52✔
628

629
structured_control_flow::ControlFlowNode* Users::scope(User* user) {
235✔
630
    if (auto data_node = dynamic_cast<data_flow::DataFlowNode*>(user->element())) {
235!
631
        return static_cast<structured_control_flow::Block*>(data_node->get_parent().get_parent());
229✔
632
    } else if (auto memlet = dynamic_cast<data_flow::Memlet*>(user->element())) {
6!
633
        return static_cast<structured_control_flow::Block*>(memlet->get_parent().get_parent());
×
634
    } else if (auto transition = dynamic_cast<structured_control_flow::Transition*>(user->element())) {
6!
635
        return &transition->parent();
6✔
636
    } else {
637
        auto user_element = dynamic_cast<structured_control_flow::ControlFlowNode*>(user->element());
×
638
        assert(user_element != nullptr && "Users::scope: User element is not a ControlFlowNode");
×
639
        return user_element;
×
640
    }
641
}
235✔
642

643
const std::unordered_set<User*> Users::all_uses_between(User& user1, User& user2) {
7✔
644
    std::unordered_set<User*> uses;
7✔
645
    std::unordered_set<User*> visited;
7✔
646
    std::list<User*> queue = {&user2};
7!
647
    while (!queue.empty()) {
40✔
648
        auto current = queue.front();
33✔
649
        queue.pop_front();
33✔
650

651
        // Stop conditions
652
        if (current == &user1) {
33✔
653
            continue;
7✔
654
        }
655

656
        if (visited.find(current) != visited.end()) {
26!
UNCOV
657
            continue;
×
658
        }
659
        visited.insert(current);
26!
660

661
        if (current != &user1 && current != &user2 && current->use() != Use::NOP) {
26!
662
            uses.insert(current);
3!
663
        }
3✔
664

665
        // Extend search
666
        // Backward search to utilize domination user1 over user
667
        auto [eb, ee] = boost::in_edges(current->vertex_, this->graph_);
26!
668
        auto edges = std::ranges::subrange(eb, ee);
52✔
669
        for (auto edge : edges) {
52!
670
            auto v = boost::source(edge, this->graph_);
26!
671
            queue.push_back(this->users_.at(v).get());
26!
672
        }
673
    }
674

675
    return uses;
7✔
676
};
7!
677

678
const std::unordered_set<User*> Users::all_uses_after(User& user) {
×
679
    std::unordered_set<User*> uses;
×
680
    std::unordered_set<User*> visited;
×
681
    std::list<User*> queue = {&user};
×
682
    while (!queue.empty()) {
×
683
        auto current = queue.front();
×
684
        queue.pop_front();
×
685
        if (visited.find(current) != visited.end()) {
×
686
            continue;
×
687
        }
688
        visited.insert(current);
×
689

690
        if (current != &user && current->use() != Use::NOP) {
×
691
            uses.insert(current);
×
692
        }
×
693

694
        // Extend search
695
        auto [eb, ee] = boost::out_edges(current->vertex_, this->graph_);
×
696
        auto edges = std::ranges::subrange(eb, ee);
×
697
        for (auto edge : edges) {
×
698
            auto v = boost::target(edge, this->graph_);
×
699
            queue.push_back(this->users_.at(v).get());
×
700
        }
701
    }
702

703
    return uses;
×
704
};
×
705

706
const std::vector<std::string> Users::all_containers_in_order() {
×
707
    std::unordered_set<std::string> unique_containers;
×
708
    std::vector<std::string> containers;
×
709

710
    // BFS traversal
711
    std::unordered_set<User*> visited;
×
712
    std::list<User*> queue = {this->source_};
×
713
    while (!queue.empty()) {
×
714
        auto current = queue.front();
×
715
        queue.pop_front();
×
716
        if (visited.find(current) != visited.end()) {
×
717
            continue;
×
718
        }
719
        visited.insert(current);
×
720

721
        if ((current->container() != "" || current->use() != Use::NOP) &&
×
722
            unique_containers.find(current->container()) == unique_containers.end()) {
×
723
            unique_containers.insert(current->container());
×
724
            containers.push_back(current->container());
×
725
        }
×
726

727
        auto [eb, ee] = boost::out_edges(current->vertex_, this->graph_);
×
728
        auto edges = std::ranges::subrange(eb, ee);
×
729
        for (auto edge : edges) {
×
730
            auto v = boost::target(edge, this->graph_);
×
731
            queue.push_back(this->users_.at(v).get());
×
732
        }
733
    }
734
    return containers;
×
735
}
×
736

737
UsersView::UsersView(Users& users, const structured_control_flow::ControlFlowNode& node) : users_(users) {
250✔
738
    this->entry_ = users.entries_.at(&node);
250!
739
    this->exit_ = users.exits_.at(&node);
250!
740

741
    // Collect sub users
742
    std::unordered_set<User*> visited;
250✔
743
    std::list<User*> queue = {this->exit_};
250!
744
    while (!queue.empty()) {
3,251✔
745
        auto current = queue.front();
3,001✔
746
        queue.pop_front();
3,001✔
747

748
        // Stop conditions
749
        if (current == this->entry_) {
3,001✔
750
            continue;
250✔
751
        }
752

753
        if (visited.find(current) != visited.end()) {
2,751!
754
            continue;
254✔
755
        }
756
        visited.insert(current);
2,497!
757

758
        this->sub_users_.insert(current);
2,497!
759

760
        // Extend search
761
        // Backward search to utilize domination user1 over user
762
        auto [eb, ee] = boost::in_edges(current->vertex_, users.graph_);
2,497!
763
        auto edges = std::ranges::subrange(eb, ee);
4,994✔
764
        for (auto edge : edges) {
5,248!
765
            auto v = boost::source(edge, users.graph_);
2,751!
766
            queue.push_back(users.users_.at(v).get());
2,751!
767
        }
768
    }
769
};
250✔
770

771
std::vector<User*> UsersView::uses() const {
81✔
772
    std::vector<User*> us;
81✔
773
    for (auto& user : this->sub_users_) {
990✔
774
        if (user->use() == Use::NOP) {
909✔
775
            continue;
379✔
776
        }
777
        us.push_back(user);
530!
778
    }
779

780
    return us;
81✔
781
};
81!
782

783
std::vector<User*> UsersView::uses(const std::string& container) const {
246✔
784
    std::vector<User*> us;
246✔
785
    for (auto& user : this->sub_users_) {
3,486✔
786
        if (user->container() != container) {
3,240✔
787
            continue;
2,685✔
788
        }
789
        if (user->use() == Use::NOP) {
555!
790
            continue;
×
791
        }
792
        us.push_back(user);
555!
793
    }
794

795
    return us;
246✔
796
};
246!
797

798
std::vector<User*> UsersView::writes() const {
2✔
799
    std::vector<User*> us;
2✔
800
    for (auto& user : this->sub_users_) {
18✔
801
        if (user->use() != Use::WRITE) {
16✔
802
            continue;
14✔
803
        }
804
        us.push_back(user);
2!
805
    }
806

807
    return us;
2✔
808
};
2!
809

810
std::vector<User*> UsersView::writes(const std::string& container) const {
137✔
811
    std::vector<User*> us;
137✔
812
    for (auto& user : this->sub_users_) {
2,881✔
813
        if (user->container() != container) {
2,744✔
814
            continue;
2,280✔
815
        }
816
        if (user->use() != Use::WRITE) {
464✔
817
            continue;
275✔
818
        }
819
        us.push_back(user);
189!
820
    }
821

822
    return us;
137✔
823
};
137!
824

825
std::vector<User*> UsersView::reads() const {
117✔
826
    std::vector<User*> us;
117✔
827
    for (auto& user : this->sub_users_) {
1,272✔
828
        if (user->use() != Use::READ) {
1,155✔
829
            continue;
670✔
830
        }
831
        us.push_back(user);
485!
832
    }
833

834
    return us;
117✔
835
};
117!
836

837
std::vector<User*> UsersView::reads(const std::string& container) const {
215✔
838
    std::vector<User*> us;
215✔
839
    for (auto& user : this->sub_users_) {
3,339✔
840
        if (user->container() != container) {
3,124✔
841
            continue;
2,623✔
842
        }
843
        if (user->use() != Use::READ) {
501✔
844
            continue;
106✔
845
        }
846
        us.push_back(user);
395!
847
    }
848

849
    return us;
215✔
850
};
215!
851

852
std::vector<User*> UsersView::views() const {
×
853
    std::vector<User*> us;
×
854
    for (auto& user : this->sub_users_) {
×
855
        if (user->use() != Use::VIEW) {
×
856
            continue;
×
857
        }
858
        us.push_back(user);
×
859
    }
860

861
    return us;
×
862
};
×
863

864
std::vector<User*> UsersView::views(const std::string& container) const {
27✔
865
    std::vector<User*> us;
27✔
866
    for (auto& user : this->sub_users_) {
329✔
867
        if (user->container() != container) {
302✔
868
            continue;
270✔
869
        }
870
        if (user->use() != Use::VIEW) {
32✔
871
            continue;
29✔
872
        }
873
        us.push_back(user);
3!
874
    }
875

876
    return us;
27✔
877
};
27!
878

879
std::vector<User*> UsersView::moves() const {
2✔
880
    std::vector<User*> us;
2✔
881
    for (auto& user : this->sub_users_) {
20✔
882
        if (user->use() != Use::MOVE) {
18✔
883
            continue;
16✔
884
        }
885
        us.push_back(user);
2!
886
    }
887

888
    return us;
2✔
889
};
2!
890

891
std::vector<User*> UsersView::moves(const std::string& container) const {
27✔
892
    std::vector<User*> us;
27✔
893
    for (auto& user : this->sub_users_) {
329✔
894
        if (user->container() != container) {
302✔
895
            continue;
270✔
896
        }
897
        if (user->use() != Use::MOVE) {
32✔
898
            continue;
29✔
899
        }
900
        us.push_back(user);
3!
901
    }
902

903
    return us;
27✔
904
};
27!
905

906
std::unordered_set<User*> UsersView::all_uses_between(User& user1, User& user2) {
×
907
    assert(this->sub_users_.find(&user1) != this->sub_users_.end());
×
908
    assert(this->sub_users_.find(&user2) != this->sub_users_.end());
×
909

910
    std::unordered_set<User*> uses;
×
911
    std::unordered_set<User*> visited;
×
912
    std::list<User*> queue = {&user1};
×
913
    while (!queue.empty()) {
×
914
        auto current = queue.front();
×
915
        queue.pop_front();
×
916
        if (visited.find(current) != visited.end()) {
×
917
            continue;
×
918
        }
919
        visited.insert(current);
×
920

921
        if (current != &user1 && current != &user2 && current->use() != Use::NOP) {
×
922
            uses.insert(current);
×
923
        }
×
924

925
        // Stop conditions
926
        if (current == exit_) {
×
927
            continue;
×
928
        }
929

930
        if (current == &user2) {
×
931
            continue;
×
932
        }
933

934
        // Extend search
935
        auto [eb, ee] = boost::out_edges(current->vertex_, this->users_.graph_);
×
936
        auto edges = std::ranges::subrange(eb, ee);
×
937
        for (auto edge : edges) {
×
938
            auto v = boost::target(edge, this->users_.graph_);
×
939
            queue.push_back(this->users_.users_.at(v).get());
×
940
        }
941
    }
942

943
    return uses;
×
944
};
×
945

946
std::unordered_set<User*> UsersView::all_uses_after(User& user) {
×
947
    assert(this->sub_users_.find(&user) != this->sub_users_.end());
×
948

949
    std::unordered_set<User*> uses;
×
950
    std::unordered_set<User*> visited;
×
951
    std::list<User*> queue = {&user};
×
952
    while (!queue.empty()) {
×
953
        auto current = queue.front();
×
954
        queue.pop_front();
×
955
        if (visited.find(current) != visited.end()) {
×
956
            continue;
×
957
        }
958
        visited.insert(current);
×
959

960
        if (current != &user && current->use() != Use::NOP) {
×
961
            uses.insert(current);
×
962
        }
×
963

964
        if (current == exit_) {
×
965
            continue;
×
966
        }
967

968
        // Extend search
969
        auto [eb, ee] = boost::out_edges(current->vertex_, this->users_.graph_);
×
970
        auto edges = std::ranges::subrange(eb, ee);
×
971
        for (auto edge : edges) {
×
972
            auto v = boost::target(edge, this->users_.graph_);
×
973
            queue.push_back(this->users_.users_.at(v).get());
×
974
        }
975
    }
976

977
    return uses;
×
978
};
×
979

980
bool Users::
981
    has_user(const std::string& container, Element* element, Use use, bool is_init, bool is_condition, bool is_update) {
208✔
982
    UserProps key{container, element, use, is_init, is_condition, is_update};
208✔
983
    if (this->users_lookup_.find(key) == this->users_lookup_.end()) {
208!
984
        return false;
156✔
985
    }
986
    return true;
52✔
987
}
208✔
988

989
User* Users::
990
    get_user(const std::string& container, Element* element, Use use, bool is_init, bool is_condition, bool is_update) {
808✔
991
    UserProps key{container, element, use, is_init, is_condition, is_update};
808✔
992
    return this->users_lookup_.at(key);
808!
993
}
808✔
994

995
void Users::add_user(std::unique_ptr<User> user) {
1,486✔
996
    auto vertex = user->vertex_;
1,486✔
997
    this->users_.insert({vertex, std::move(user)});
1,486!
998

999
    auto user_ptr = this->users_.at(vertex).get();
1,486✔
1000
    bool is_init = false;
1,486✔
1001
    bool is_condition = false;
1,486✔
1002
    bool is_update = false;
1,486✔
1003
    if (auto for_user = dynamic_cast<ForUser*>(user_ptr)) {
1,486!
1004
        auto for_loop = dynamic_cast<structured_control_flow::StructuredLoop*>(user_ptr->element());
647!
1005
        if (for_loop == nullptr) {
647!
1006
            throw std::invalid_argument("Invalid user type");
×
1007
        }
1008
        if (for_user->is_init()) {
647✔
1009
            is_init = true;
143✔
1010
        } else if (for_user->is_condition()) {
647✔
1011
            is_condition = true;
246✔
1012
        } else if (for_user->is_update()) {
504!
1013
            is_update = true;
258✔
1014
        } else {
258✔
1015
            throw std::invalid_argument("Invalid user type");
×
1016
        }
1017
    }
647✔
1018

1019
    UserProps key{user_ptr->container(), user_ptr->element(), user_ptr->use(), is_init, is_condition, is_update};
1,486✔
1020
    this->users_lookup_.insert({key, user_ptr});
1,486!
1021
}
1,486✔
1022

1023
std::unordered_set<std::string> Users::locals(structured_control_flow::ControlFlowNode& node) {
20✔
1024
    auto& sdfg = this->sdfg_;
20✔
1025

1026
    // Locals have no uses outside of the node
1027
    // We can check this by comparing the number of uses of the container in the view and the total
1028
    // number of uses of the container in the users map.
1029
    std::unordered_set<std::string> locals;
20✔
1030
    UsersView view(*this, node);
20!
1031
    for (auto& user : view.uses()) {
73!
1032
        if (!sdfg.is_transient(user->container())) {
53!
1033
            continue;
21✔
1034
        }
1035
        if (view.uses(user->container()).size() == this->uses(user->container()).size()) {
32!
1036
            locals.insert(user->container());
12!
1037
        }
12✔
1038
    }
1039

1040
    return locals;
20✔
1041
};
20!
1042

1043
const std::vector<std::string> UsersView::all_containers_in_order() {
×
1044
    std::unordered_set<std::string> unique_containers;
×
1045
    std::vector<std::string> containers;
×
1046

1047
    // BFS traversal
1048
    std::unordered_set<User*> visited;
×
1049
    std::list<User*> queue = {this->entry_};
×
1050
    while (!queue.empty()) {
×
1051
        auto current = queue.front();
×
1052
        queue.pop_front();
×
1053
        if (visited.find(current) != visited.end()) {
×
1054
            continue;
×
1055
        }
1056
        visited.insert(current);
×
1057

1058
        if ((current->container() != "" || current->use() != Use::NOP) &&
×
1059
            unique_containers.find(current->container()) == unique_containers.end()) {
×
1060
            unique_containers.insert(current->container());
×
1061
            containers.push_back(current->container());
×
1062
        }
×
1063

1064
        if (current == this->exit_) {
×
1065
            continue;
×
1066
        }
1067

1068
        auto [eb, ee] = boost::out_edges(current->vertex_, this->users_.graph_);
×
1069
        auto edges = std::ranges::subrange(eb, ee);
×
1070
        for (auto edge : edges) {
×
1071
            auto v = boost::target(edge, this->users_.graph_);
×
1072
            queue.push_back(this->users_.users_.at(v).get());
×
1073
        }
1074
    }
1075

1076
    return containers;
×
1077
}
×
1078

1079
} // namespace analysis
1080
} // 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