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

daisytuner / docc / 22287993212

22 Feb 2026 11:45PM UTC coverage: 64.63% (-0.1%) from 64.743%
22287993212

Pull #537

github

web-flow
Merge d546ea4e1 into d39f6af54
Pull Request #537: Adds memory management to frontend

48 of 51 new or added lines in 4 files covered. (94.12%)

62 existing lines in 4 files now uncovered.

23672 of 36627 relevant lines covered (64.63%)

357.5 hits per line

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

78.46
/sdfg/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)
25
    : vertex_(vertex), container_(container), use_(use), element_(element) {
14,398✔
26

27
      };
14,398✔
28

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

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

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

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

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

58
    this->subsets_cached_ = true;
772✔
59
    return this->subsets_;
772✔
60
};
3,917✔
61

62
ForUser::ForUser(
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) {
3,696✔
72

73
      };
3,696✔
74

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

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

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

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

89
        if (auto access_node = dynamic_cast<data_flow::AccessNode*>(node)) {
2,750✔
90
            if (!symbolic::is_pointer(symbolic::symbol(access_node->data()))) {
1,919✔
91
                if (dataflow.in_degree(*node) > 0) {
1,919✔
92
                    Use use = Use::WRITE;
831✔
93

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

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

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

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

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

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

149
        for (auto& oedge : dataflow.out_edges(*node)) {
2,750✔
150
            std::unordered_set<std::string> used;
1,943✔
151
            for (auto dim : oedge.subset()) {
2,094✔
152
                for (auto atom : symbolic::atoms(dim)) {
2,094✔
153
                    if (used.find(atom->get_name()) != used.end()) {
1,982✔
154
                        continue;
10✔
155
                    }
10✔
156
                    used.insert(atom->get_name());
1,972✔
157

158
                    auto v = boost::add_vertex(this->graph_);
1,972✔
159
                    this->add_user(std::make_unique<User>(v, atom->get_name(), &oedge, Use::READ));
1,972✔
160
                    if (last != boost::graph_traits<graph::Graph>::null_vertex()) {
1,972✔
161
                        boost::add_edge(last, v, this->graph_);
1,898✔
162
                    } else {
1,898✔
163
                        first = v;
74✔
164
                    }
74✔
165
                    last = v;
1,972✔
166
                }
1,972✔
167
            }
2,094✔
168
        }
1,943✔
169
    }
2,750✔
170

171
    return {first, last};
1,004✔
172
};
1,004✔
173

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

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

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

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

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

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

204
        graph::Vertex current = s;
1,369✔
205
        for (size_t i = 0; i < sequence_stmt->size(); i++) {
3,235✔
206
            auto child = sequence_stmt->at(i);
1,872✔
207

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

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

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

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

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

239
                boost::add_edge(current, v, this->graph_);
156✔
240
                current = v;
156✔
241
            }
156✔
242
        }
1,866✔
243

244
        if (current == boost::graph_traits<graph::Graph>::null_vertex()) {
1,369✔
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_);
1,369✔
250
        this->users_.insert({t, std::make_unique<User>(t, "", sequence_stmt, Use::NOP)});
1,369✔
251
        boost::add_edge(current, t, this->graph_);
1,369✔
252
        this->exits_.insert({sequence_stmt, this->users_.at(t).get()});
1,369✔
253

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

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

263
        std::unordered_set<std::string> used;
58✔
264
        for (size_t i = 0; i < if_else_stmt->size(); i++) {
140✔
265
            auto condition = if_else_stmt->at(i).second;
82✔
266
            for (auto atom : symbolic::atoms(condition)) {
83✔
267
                if (used.find(atom->get_name()) != used.end()) {
83✔
268
                    continue;
18✔
269
                }
18✔
270
                used.insert(atom->get_name());
65✔
271

272
                auto v = boost::add_vertex(this->graph_);
65✔
273

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

276
                boost::add_edge(last, v, this->graph_);
65✔
277
                last = v;
65✔
278
            }
65✔
279
        }
82✔
280

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

296
        // Forward edge: Potentially missing else case
297
        if (!if_else_stmt->is_complete()) {
58✔
298
            if (t == boost::graph_traits<graph::Graph>::null_vertex()) {
34✔
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_);
34✔
304
        }
34✔
305

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

396
        return {s, t};
755✔
397
    } else if (auto cont_stmt = dynamic_cast<structured_control_flow::Continue*>(&node)) {
755✔
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)) {
11✔
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)) {
6✔
412
        if (!ret_stmt->is_data() || ret_stmt->data().empty()) {
6✔
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 {
6✔
419
            auto v = boost::add_vertex(this->graph_);
6✔
420
            this->add_user(std::make_unique<User>(v, ret_stmt->data(), ret_stmt, Use::READ));
6✔
421
            this->entries_.insert({ret_stmt, this->users_.at(v).get()});
6✔
422
            this->exits_.insert({ret_stmt, this->users_.at(v).get()});
6✔
423
            return {v, boost::graph_traits<graph::Graph>::null_vertex()};
6✔
424
        }
6✔
425
    }
6✔
426

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

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

433
      };
492✔
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) {
492✔
441
    users_.clear();
492✔
442
    graph_.clear();
492✔
443
    source_ = nullptr;
492✔
444
    sink_ = nullptr;
492✔
445
    this->entries_.clear();
492✔
446
    this->exits_.clear();
492✔
447
    this->users_lookup_.clear();
492✔
448

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

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

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

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

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

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

503
    // Collect sub structures
504
    for (auto& entry : this->users_) {
14,393✔
505
        auto container = entry.second->container();
14,393✔
506
        if (entry.second->use() == Use::NOP) {
14,393✔
507
            continue;
6,450✔
508
        }
6,450✔
509
        if (container == "") {
7,943✔
510
            continue;
×
511
        }
×
512

513
        switch (entry.second->use()) {
7,943✔
514
            case Use::READ: {
5,413✔
515
                this->reads_[container].push_back(entry.second.get());
5,413✔
516
                break;
5,413✔
517
            }
×
518
            case Use::WRITE: {
2,459✔
519
                this->writes_[container].push_back(entry.second.get());
2,459✔
520
                break;
2,459✔
521
            }
×
522
            case Use::VIEW: {
33✔
523
                this->views_[container].push_back(entry.second.get());
33✔
524
                break;
33✔
525
            }
×
526
            case Use::MOVE: {
38✔
527
                this->moves_[container].push_back(entry.second.get());
38✔
528
                break;
38✔
529
            }
×
530
            default:
×
531
                break;
×
532
        }
7,943✔
533
    }
7,943✔
534
};
492✔
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 {
2,004✔
549
    std::list<User*> us;
2,004✔
550
    for (auto& entry : this->users_) {
131,751✔
551
        if (entry.second->container() != container) {
131,751✔
552
            continue;
121,407✔
553
        }
121,407✔
554
        if (entry.second->use() == Use::NOP) {
10,344✔
555
            continue;
×
556
        }
×
557
        us.push_back(entry.second.get());
10,344✔
558
    }
10,344✔
559

560
    return us;
2,004✔
561
};
2,004✔
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 {
3✔
566
    std::list<User*> us;
3✔
567
    for (auto& entry : this->users_) {
29✔
568
        if (entry.second->use() != Use::WRITE) {
29✔
569
            continue;
24✔
570
        }
24✔
571
        us.push_back(entry.second.get());
5✔
572
    }
5✔
573

574
    return us;
3✔
575
};
3✔
576

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

579
size_t Users::num_writes(const std::string& container) const { return this->writes(container).size(); };
7✔
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); };
2,477✔
594

595
size_t Users::num_reads(const std::string& container) const { return this->reads(container).size(); };
1,880✔
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); };
444✔
610

611
size_t Users::num_views(const std::string& container) const { return this->views(container).size(); };
429✔
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); };
486✔
626

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

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

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

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

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

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

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

675
    return uses;
11✔
676
};
11✔
677

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

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

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

703
    return uses;
8✔
704
};
8✔
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) {
1,691✔
738
    this->entry_ = users.entries_.at(&node);
1,691✔
739
    this->exit_ = users.exits_.at(&node);
1,691✔
740

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

748
        // Stop conditions
749
        if (current == this->entry_) {
38,751✔
750
            continue;
1,691✔
751
        }
1,691✔
752

753
        if (visited.find(current) != visited.end()) {
37,060✔
754
            continue;
4,370✔
755
        }
4,370✔
756
        visited.insert(current);
32,690✔
757

758
        this->sub_users_.insert(current);
32,690✔
759

760
        // Extend search
761
        // Backward search to utilize domination user1 over user
762
        auto [eb, ee] = boost::in_edges(current->vertex_, users.graph_);
32,690✔
763
        auto edges = std::ranges::subrange(eb, ee);
32,690✔
764
        for (auto edge : edges) {
37,060✔
765
            auto v = boost::source(edge, users.graph_);
37,060✔
766
            queue.push_back(users.users_.at(v).get());
37,060✔
767
        }
37,060✔
768
    }
32,690✔
769
};
1,691✔
770

771
std::vector<User*> UsersView::uses() const {
634✔
772
    std::vector<User*> us;
634✔
773
    for (auto& user : this->sub_users_) {
11,489✔
774
        if (user->use() == Use::NOP) {
11,489✔
775
            continue;
4,226✔
776
        }
4,226✔
777
        us.push_back(user);
7,263✔
778
    }
7,263✔
779

780
    return us;
634✔
781
};
634✔
782

783
std::vector<User*> UsersView::uses(const std::string& container) const {
2,916✔
784
    std::vector<User*> us;
2,916✔
785
    for (auto& user : this->sub_users_) {
85,036✔
786
        if (user->container() != container) {
85,036✔
787
            continue;
73,122✔
788
        }
73,122✔
789
        if (user->use() == Use::NOP) {
11,914✔
790
            continue;
×
791
        }
×
792
        us.push_back(user);
11,914✔
793
    }
11,914✔
794

795
    return us;
2,916✔
796
};
2,916✔
797

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

807
    return us;
72✔
808
};
72✔
809

810
std::vector<User*> UsersView::writes(const std::string& container) const {
2,067✔
811
    std::vector<User*> us;
2,067✔
812
    for (auto& user : this->sub_users_) {
44,604✔
813
        if (user->container() != container) {
44,604✔
814
            continue;
34,755✔
815
        }
34,755✔
816
        if (user->use() != Use::WRITE) {
9,849✔
817
            continue;
6,374✔
818
        }
6,374✔
819
        us.push_back(user);
3,475✔
820
    }
3,475✔
821

822
    return us;
2,067✔
823
};
2,067✔
824

825
std::vector<User*> UsersView::reads() const {
641✔
826
    std::vector<User*> us;
641✔
827
    for (auto& user : this->sub_users_) {
13,989✔
828
        if (user->use() != Use::READ) {
13,989✔
829
            continue;
6,814✔
830
        }
6,814✔
831
        us.push_back(user);
7,175✔
832
    }
7,175✔
833

834
    return us;
641✔
835
};
641✔
836

837
std::vector<User*> UsersView::reads(const std::string& container) const {
1,604✔
838
    std::vector<User*> us;
1,604✔
839
    for (auto& user : this->sub_users_) {
44,310✔
840
        if (user->container() != container) {
44,310✔
841
            continue;
38,939✔
842
        }
38,939✔
843
        if (user->use() != Use::READ) {
5,371✔
844
            continue;
854✔
845
        }
854✔
846
        us.push_back(user);
4,517✔
847
    }
4,517✔
848

849
    return us;
1,604✔
850
};
1,604✔
851

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

861
    return us;
40✔
862
};
40✔
863

864
std::vector<User*> UsersView::views(const std::string& container) const {
146✔
865
    std::vector<User*> us;
146✔
866
    for (auto& user : this->sub_users_) {
2,403✔
867
        if (user->container() != container) {
2,403✔
868
            continue;
2,106✔
869
        }
2,106✔
870
        if (user->use() != Use::VIEW) {
297✔
871
            continue;
293✔
872
        }
293✔
873
        us.push_back(user);
4✔
874
    }
4✔
875

876
    return us;
146✔
877
};
146✔
878

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

888
    return us;
42✔
889
};
42✔
890

891
std::vector<User*> UsersView::moves(const std::string& container) const {
158✔
892
    std::vector<User*> us;
158✔
893
    for (auto& user : this->sub_users_) {
2,585✔
894
        if (user->container() != container) {
2,585✔
895
            continue;
2,258✔
896
        }
2,258✔
897
        if (user->use() != Use::MOVE) {
327✔
898
            continue;
322✔
899
        }
322✔
900
        us.push_back(user);
5✔
901
    }
5✔
902

903
    return us;
158✔
904
};
158✔
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) {
7✔
947
    assert(this->sub_users_.find(&user) != this->sub_users_.end());
7✔
948

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

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

964
        if (current == exit_) {
24✔
965
            continue;
7✔
966
        }
7✔
967

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

977
    return uses;
7✔
978
};
7✔
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
    }
156✔
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) {
7,866✔
991
    UserProps key{container, element, use, is_init, is_condition, is_update};
7,866✔
992
    return this->users_lookup_.at(key);
7,866✔
993
}
7,866✔
994

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

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

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

1023
std::unordered_set<std::string> Users::locals(structured_control_flow::ControlFlowNode& node) {
122✔
1024
    auto& sdfg = this->sdfg_;
122✔
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;
122✔
1030
    UsersView view(*this, node);
122✔
1031
    for (auto& user : view.uses()) {
1,621✔
1032
        if (!sdfg.is_transient(user->container())) {
1,621✔
1033
            continue;
468✔
1034
        }
468✔
1035
        if (view.uses(user->container()).size() == this->uses(user->container()).size()) {
1,153✔
1036
            locals.insert(user->container());
660✔
1037
        }
660✔
1038
    }
1,153✔
1039

1040
    return locals;
122✔
1041
};
122✔
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