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

daisytuner / docc / 23543227962

25 Mar 2026 01:22PM UTC coverage: 64.841% (+0.5%) from 64.295%
23543227962

Pull #609

github

web-flow
Merge 4c67e4cfe into d8e714075
Pull Request #609: Enable Loop Collapse for Offloading

78 of 105 new or added lines in 9 files covered. (74.29%)

293 existing lines in 2 files now uncovered.

26983 of 41614 relevant lines covered (64.84%)

407.97 hits per line

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

31.31
/sdfg/src/codegen/language_extensions/cpp_language_extension.cpp
1
#include "sdfg/codegen/language_extensions/cpp_language_extension.h"
2

3
#include "sdfg/codegen/utils.h"
4
#include "sdfg/data_flow/tasklet.h"
5

6
namespace sdfg {
7
namespace codegen {
8

9
std::string CPPLanguageExtension::primitive_type(const types::PrimitiveType prim_type) {
40✔
10
    switch (prim_type) {
40✔
11
        case types::PrimitiveType::Void:
3✔
12
            return "void";
3✔
13
        case types::PrimitiveType::Bool:
1✔
14
            return "bool";
1✔
15
        case types::PrimitiveType::Int8:
1✔
16
            return "int8_t";
1✔
17
        case types::PrimitiveType::Int16:
1✔
18
            return "int16_t";
1✔
19
        case types::PrimitiveType::Int32:
6✔
20
            return "int32_t";
6✔
21
        case types::PrimitiveType::Int64:
1✔
22
            return "int64_t";
1✔
23
        case types::PrimitiveType::Int128:
×
24
            return "__int128";
×
25
        case types::PrimitiveType::UInt8:
4✔
26
            return "uint8_t";
4✔
27
        case types::PrimitiveType::UInt16:
1✔
28
            return "uint16_t";
1✔
29
        case types::PrimitiveType::UInt32:
1✔
30
            return "uint32_t";
1✔
31
        case types::PrimitiveType::UInt64:
4✔
32
            return "uint64_t";
4✔
33
        case types::PrimitiveType::UInt128:
×
34
            return "unsigned __int128";
×
35
        case types::PrimitiveType::Half:
×
36
            return "_Float16";
×
37
        case types::PrimitiveType::BFloat:
×
38
            return "__bf16";
×
39
        case types::PrimitiveType::Float:
16✔
40
            return "float";
16✔
41
        case types::PrimitiveType::Double:
1✔
42
            return "double";
1✔
43
        case types::PrimitiveType::X86_FP80:
×
44
            return "long double";
×
45
        case types::PrimitiveType::FP128:
×
46
            return "__float128";
×
47
        case types::PrimitiveType::PPC_FP128:
×
48
            return "__float128";
×
49
    }
40✔
50

51
    throw std::runtime_error("Unknown primitive type");
×
52
};
40✔
53

54
std::string CPPLanguageExtension::
55
    declaration(const std::string& name, const types::IType& type, bool use_initializer, bool use_alignment) {
57✔
56
    std::stringstream val;
57✔
57

58
    if (auto scalar_type = dynamic_cast<const types::Scalar*>(&type)) {
57✔
59
        val << primitive_type(scalar_type->primitive_type());
28✔
60
        val << " ";
28✔
61
        val << name;
28✔
62
    } else if (auto array_type = dynamic_cast<const types::Array*>(&type)) {
29✔
63
        auto& element_type = array_type->element_type();
10✔
64
        val << declaration(name + "[" + this->expression(array_type->num_elements()) + "]", element_type);
10✔
65
    } else if (auto pointer_type = dynamic_cast<const types::Pointer*>(&type)) {
19✔
66
        if (pointer_type->has_pointee_type()) {
15✔
67
            const types::IType& pointee = pointer_type->pointee_type();
13✔
68

69
            const bool pointee_is_function_or_array = dynamic_cast<const types::Function*>(&pointee) ||
13✔
70
                                                      dynamic_cast<const types::Array*>(&pointee);
13✔
71

72
            // Parenthesise *only* when it is needed to bind tighter than [] or ()
73
            std::string decorated = pointee_is_function_or_array ? "(*" + name + ")" : "*" + name;
13✔
74

75
            val << declaration(decorated, pointee);
13✔
76
        } else {
13✔
77
            val << "void*";
2✔
78
            val << " " << name;
2✔
79
        }
2✔
80
    } else if (auto ref_type = dynamic_cast<const Reference*>(&type)) {
15✔
81
        val << declaration("&" + name, ref_type->reference_type());
×
82
    } else if (auto structure_type = dynamic_cast<const types::Structure*>(&type)) {
4✔
83
        val << structure_type->name();
4✔
84
        val << " ";
4✔
85
        val << name;
4✔
86
    } else if (auto function_type = dynamic_cast<const types::Function*>(&type)) {
4✔
87
        std::stringstream params;
×
88
        for (size_t i = 0; i < function_type->num_params(); ++i) {
×
89
            params << declaration("", function_type->param_type(symbolic::integer(i)));
×
90
            if (i + 1 < function_type->num_params()) params << ", ";
×
91
        }
×
92
        if (function_type->is_var_arg()) {
×
93
            if (function_type->num_params() > 0) {
×
94
                params << ", ";
×
95
            }
×
96
            params << "...";
×
97
        }
×
98

99
        const std::string fun_name = name + "(" + params.str() + ")";
×
100
        val << declaration(fun_name, function_type->return_type());
×
101
    } else {
×
102
        throw std::runtime_error("Unknown declaration type");
×
103
    }
×
104

105
    if (use_alignment && type.alignment() > 0) {
57✔
106
        val << " __attribute__((aligned(" << type.alignment() << ")))";
×
107
    }
×
108
    if (use_initializer && !type.initializer().empty()) {
57✔
109
        val << " = " << type.initializer();
×
110
    }
×
111

112
    return val.str();
57✔
113
};
57✔
114

115
std::string CPPLanguageExtension::type_cast(const std::string& name, const types::IType& type) {
6✔
116
    std::stringstream val;
6✔
117

118
    val << "reinterpret_cast";
6✔
119
    val << "<";
6✔
120
    val << declaration("", type);
6✔
121
    val << ">";
6✔
122
    val << "(" << name << ")";
6✔
123

124
    return val.str();
6✔
125
};
6✔
126

127
std::string CPPLanguageExtension::subset(const types::IType& type, const data_flow::Subset& sub) {
13✔
128
    if (sub.empty()) {
13✔
129
        return "";
3✔
130
    }
3✔
131

132
    if (dynamic_cast<const types::Scalar*>(&type)) {
10✔
133
        return "";
×
134
    } else if (auto array_type = dynamic_cast<const types::Array*>(&type)) {
10✔
135
        std::string subset_str = "[" + this->expression(sub.at(0)) + "]";
5✔
136

137
        if (sub.size() > 1) {
5✔
138
            data_flow::Subset element_subset(sub.begin() + 1, sub.end());
×
139
            auto& element_type = array_type->element_type();
×
140
            return subset_str + subset(element_type, element_subset);
×
141
        } else {
5✔
142
            return subset_str;
5✔
143
        }
5✔
144
    } else if (auto pointer_type = dynamic_cast<const types::Pointer*>(&type)) {
5✔
145
        std::string subset_str = "[" + this->expression(sub.at(0)) + "]";
4✔
146

147
        data_flow::Subset element_subset(sub.begin() + 1, sub.end());
4✔
148
        auto& pointee_type = pointer_type->pointee_type();
4✔
149
        return subset_str + subset(pointee_type, element_subset);
4✔
150
    } else if (auto structure_type = dynamic_cast<const types::Structure*>(&type)) {
4✔
151
        auto& definition = this->function_.structure(structure_type->name());
1✔
152

153
        std::string subset_str = ".member_" + this->expression(sub.at(0));
1✔
154
        if (sub.size() > 1) {
1✔
155
            auto member = SymEngine::rcp_dynamic_cast<const SymEngine::Integer>(sub.at(0));
×
156
            auto& member_type = definition.member_type(member);
×
157
            data_flow::Subset element_subset(sub.begin() + 1, sub.end());
×
158
            return subset_str + subset(member_type, element_subset);
×
159
        } else {
1✔
160
            return subset_str;
1✔
161
        }
1✔
162
    }
1✔
163

164
    throw std::invalid_argument("Invalid subset type");
×
165
};
10✔
166

167
std::string CPPLanguageExtension::expression(const symbolic::Expression expr) {
34✔
168
    CPPSymbolicPrinter printer(this->function_, this->external_prefix_);
34✔
169
    return printer.apply(expr);
34✔
170
};
34✔
171

172
std::string CPPLanguageExtension::access_node(const data_flow::AccessNode& node) {
6✔
173
    if (dynamic_cast<const data_flow::ConstantNode*>(&node)) {
6✔
174
        std::string name = node.data();
×
175
        if (symbolic::is_nullptr(symbolic::symbol(name))) {
×
176
            return "nullptr";
×
177
        }
×
178
        return name;
×
179
    } else {
6✔
180
        std::string name = node.data();
6✔
181
        if (this->function_.is_external(name)) {
6✔
182
            return "(&" + this->external_prefix_ + name + ")";
×
183
        }
×
184
        return name;
6✔
185
    }
6✔
186
};
6✔
187

188
std::string CPPLanguageExtension::tasklet(const data_flow::Tasklet& tasklet) {
1✔
189
    switch (tasklet.code()) {
1✔
190
        case data_flow::TaskletCode::assign:
×
191
            return tasklet.inputs().at(0);
×
192
        case data_flow::TaskletCode::fp_neg:
×
193
            return "-" + tasklet.inputs().at(0);
×
194
        case data_flow::TaskletCode::fp_add:
×
195
            return tasklet.inputs().at(0) + " + " + tasklet.inputs().at(1);
×
196
        case data_flow::TaskletCode::fp_sub:
×
197
            return tasklet.inputs().at(0) + " - " + tasklet.inputs().at(1);
×
198
        case data_flow::TaskletCode::fp_mul:
×
199
            return tasklet.inputs().at(0) + " * " + tasklet.inputs().at(1);
×
200
        case data_flow::TaskletCode::fp_div:
×
201
            return tasklet.inputs().at(0) + " / " + tasklet.inputs().at(1);
×
202
        case data_flow::TaskletCode::fp_rem:
×
203
            return "fmod(" + tasklet.inputs().at(0) + ", " + tasklet.inputs().at(1) + ")";
×
204
        case data_flow::TaskletCode::fp_fma:
1✔
205
            return tasklet.inputs().at(0) + " * " + tasklet.inputs().at(1) + " + " + tasklet.inputs().at(2);
1✔
206
        case data_flow::TaskletCode::fp_oeq:
×
207
            return tasklet.inputs().at(0) + " == " + tasklet.inputs().at(1);
×
208
        case data_flow::TaskletCode::fp_one:
×
209
            return "!std::isnan(" + tasklet.inputs().at(0) + ") && !std::isnan(" + tasklet.inputs().at(1) + ") && " +
×
210
                   tasklet.inputs().at(0) + " != " + tasklet.inputs().at(1);
×
211
        case data_flow::TaskletCode::fp_ogt:
×
212
            return tasklet.inputs().at(0) + " > " + tasklet.inputs().at(1);
×
213
        case data_flow::TaskletCode::fp_oge:
×
214
            return tasklet.inputs().at(0) + " >= " + tasklet.inputs().at(1);
×
215
        case data_flow::TaskletCode::fp_olt:
×
216
            return tasklet.inputs().at(0) + " < " + tasklet.inputs().at(1);
×
217
        case data_flow::TaskletCode::fp_ole:
×
218
            return tasklet.inputs().at(0) + " <= " + tasklet.inputs().at(1);
×
219
        case data_flow::TaskletCode::fp_ord:
×
220
            return "std::isnan(" + tasklet.inputs().at(0) + ") && std::isnan(" + tasklet.inputs().at(1) + ")";
×
221
        case data_flow::TaskletCode::fp_ueq:
×
222
            return "std::isnan(" + tasklet.inputs().at(0) + ") || std::isnan(" + tasklet.inputs().at(1) + ")" + " || " +
×
223
                   tasklet.inputs().at(0) + " == " + tasklet.inputs().at(1);
×
224
        case data_flow::TaskletCode::fp_une:
×
225
            return "std::isnan(" + tasklet.inputs().at(0) + ") || std::isnan(" + tasklet.inputs().at(1) + ")" + " || " +
×
226
                   tasklet.inputs().at(0) + " != " + tasklet.inputs().at(1);
×
227
        case data_flow::TaskletCode::fp_ugt:
×
228
            return "std::isnan(" + tasklet.inputs().at(0) + ") || std::isnan(" + tasklet.inputs().at(1) + ")" + " || " +
×
229
                   tasklet.inputs().at(0) + " > " + tasklet.inputs().at(1);
×
230
        case data_flow::TaskletCode::fp_uge:
×
231
            return "std::isnan(" + tasklet.inputs().at(0) + ") || std::isnan(" + tasklet.inputs().at(1) + ")" + " || " +
×
232
                   tasklet.inputs().at(0) + " >= " + tasklet.inputs().at(1);
×
233
        case data_flow::TaskletCode::fp_ult:
×
234
            return "std::isnan(" + tasklet.inputs().at(0) + ") || std::isnan(" + tasklet.inputs().at(1) + ")" + " || " +
×
235
                   tasklet.inputs().at(0) + " < " + tasklet.inputs().at(1);
×
236
        case data_flow::TaskletCode::fp_ule:
×
237
            return "std::isnan(" + tasklet.inputs().at(0) + ") || std::isnan(" + tasklet.inputs().at(1) + ")" + " || " +
×
238
                   tasklet.inputs().at(0) + " <= " + tasklet.inputs().at(1);
×
239
        case data_flow::TaskletCode::fp_uno:
×
240
            return "std::isnan(" + tasklet.inputs().at(0) + ") || std::isnan(" + tasklet.inputs().at(1) + ")";
×
241
        case data_flow::TaskletCode::int_add:
×
242
            return tasklet.inputs().at(0) + " + " + tasklet.inputs().at(1);
×
243
        case data_flow::TaskletCode::int_sub:
×
244
            return tasklet.inputs().at(0) + " - " + tasklet.inputs().at(1);
×
245
        case data_flow::TaskletCode::int_mul:
×
246
            return tasklet.inputs().at(0) + " * " + tasklet.inputs().at(1);
×
247
        case data_flow::TaskletCode::int_sdiv:
×
248
            return tasklet.inputs().at(0) + " / " + tasklet.inputs().at(1);
×
249
        case data_flow::TaskletCode::int_srem:
×
250
            return tasklet.inputs().at(0) + " % " + tasklet.inputs().at(1);
×
251
        case data_flow::TaskletCode::int_udiv:
×
252
            return tasklet.inputs().at(0) + " / " + tasklet.inputs().at(1);
×
253
        case data_flow::TaskletCode::int_urem:
×
254
            return tasklet.inputs().at(0) + " % " + tasklet.inputs().at(1);
×
255
        case data_flow::TaskletCode::int_and:
×
256
            return tasklet.inputs().at(0) + " & " + tasklet.inputs().at(1);
×
257
        case data_flow::TaskletCode::int_or:
×
258
            return tasklet.inputs().at(0) + " | " + tasklet.inputs().at(1);
×
259
        case data_flow::TaskletCode::int_xor:
×
260
            return tasklet.inputs().at(0) + " ^ " + tasklet.inputs().at(1);
×
261
        case data_flow::TaskletCode::int_shl:
×
262
            return tasklet.inputs().at(0) + " << " + tasklet.inputs().at(1);
×
263
        case data_flow::TaskletCode::int_lshr:
×
264
            return tasklet.inputs().at(0) + " >> " + tasklet.inputs().at(1);
×
265
        case data_flow::TaskletCode::int_ashr:
×
266
            return tasklet.inputs().at(0) + " >> " + tasklet.inputs().at(1);
×
267
        case data_flow::TaskletCode::int_smin:
×
268
            return tasklet.inputs().at(0) + " < " + tasklet.inputs().at(1) + " ? " + tasklet.inputs().at(0) + " : " +
×
269
                   tasklet.inputs().at(1);
×
270
        case data_flow::TaskletCode::int_smax:
×
271
            return tasklet.inputs().at(0) + " > " + tasklet.inputs().at(1) + " ? " + tasklet.inputs().at(0) + " : " +
×
272
                   tasklet.inputs().at(1);
×
273
        case data_flow::TaskletCode::int_scmp:
×
274
            return tasklet.inputs().at(0) + " < " + tasklet.inputs().at(1) + " ? -1 : (" + tasklet.inputs().at(0) +
×
275
                   " > " + tasklet.inputs().at(1) + " ? 1 : 0)";
×
276
        case data_flow::TaskletCode::int_umin:
×
277
            return tasklet.inputs().at(0) + " < " + tasklet.inputs().at(1) + " ? " + tasklet.inputs().at(0) + " : " +
×
278
                   tasklet.inputs().at(1);
×
279
        case data_flow::TaskletCode::int_umax:
×
280
            return tasklet.inputs().at(0) + " > " + tasklet.inputs().at(1) + " ? " + tasklet.inputs().at(0) + " : " +
×
281
                   tasklet.inputs().at(1);
×
282
        case data_flow::TaskletCode::int_ucmp:
×
283
            return tasklet.inputs().at(0) + " < " + tasklet.inputs().at(1) + " ? -1 : (" + tasklet.inputs().at(0) +
×
284
                   " > " + tasklet.inputs().at(1) + " ? 1 : 0)";
×
285
        case data_flow::TaskletCode::int_abs:
×
286
            return "(" + tasklet.inputs().at(0) + " < 0 ? -" + tasklet.inputs().at(0) + " : " + tasklet.inputs().at(0) +
×
287
                   ")";
×
288
        case data_flow::TaskletCode::int_eq:
×
289
            return tasklet.inputs().at(0) + " == " + tasklet.inputs().at(1);
×
290
        case data_flow::TaskletCode::int_ne:
×
291
            return tasklet.inputs().at(0) + " != " + tasklet.inputs().at(1);
×
292
        case data_flow::TaskletCode::int_sgt:
×
293
            return tasklet.inputs().at(0) + " > " + tasklet.inputs().at(1);
×
294
        case data_flow::TaskletCode::int_sge:
×
295
            return tasklet.inputs().at(0) + " >= " + tasklet.inputs().at(1);
×
296
        case data_flow::TaskletCode::int_slt:
×
297
            return tasklet.inputs().at(0) + " < " + tasklet.inputs().at(1);
×
298
        case data_flow::TaskletCode::int_sle:
×
299
            return tasklet.inputs().at(0) + " <= " + tasklet.inputs().at(1);
×
300
        case data_flow::TaskletCode::int_ugt:
×
301
            return tasklet.inputs().at(0) + " > " + tasklet.inputs().at(1);
×
302
        case data_flow::TaskletCode::int_uge:
×
303
            return tasklet.inputs().at(0) + " >= " + tasklet.inputs().at(1);
×
304
        case data_flow::TaskletCode::int_ult:
×
305
            return tasklet.inputs().at(0) + " < " + tasklet.inputs().at(1);
×
306
        case data_flow::TaskletCode::int_ule:
×
307
            return tasklet.inputs().at(0) + " <= " + tasklet.inputs().at(1);
×
308
    };
1✔
309
    throw std::invalid_argument("Invalid tasklet code");
×
310
};
1✔
311

312
std::string CPPLanguageExtension::zero(const types::PrimitiveType prim_type) {
×
313
    switch (prim_type) {
×
314
        case types::Void:
×
315
            throw InvalidSDFGException("No zero for void type possible");
×
316
        case types::Bool:
×
317
            return "false";
×
318
        case types::Int8:
×
319
            return "0";
×
320
        case types::Int16:
×
321
            return "0";
×
322
        case types::Int32:
×
323
            return "0";
×
324
        case types::Int64:
×
325
            return "0ll";
×
326
        case types::Int128:
×
327
            return "0";
×
328
        case types::UInt8:
×
329
            return "0u";
×
330
        case types::UInt16:
×
331
            return "0u";
×
332
        case types::UInt32:
×
333
            return "0u";
×
334
        case types::UInt64:
×
335
            return "0ull";
×
336
        case types::UInt128:
×
337
            return "0u";
×
338
        case types::Half:
×
339
            throw InvalidSDFGException("Currently unsupported");
×
340
        case types::BFloat:
×
341
            throw InvalidSDFGException("Currently unsupported");
×
342
        case types::Float:
×
343
            return "0.0f";
×
344
        case types::Double:
×
345
            return "0.0";
×
346
        case types::X86_FP80:
×
347
            return "0.0l";
×
348
        case types::FP128:
×
349
            throw InvalidSDFGException("Currently unsupported");
×
350
        case types::PPC_FP128:
×
351
            throw InvalidSDFGException("Currently unsupported");
×
352
    }
×
353
}
×
354

355
void CPPSymbolicPrinter::bvisit(const SymEngine::Infty& x) {
×
356
    if (x.is_negative_infinity())
×
357
        str_ = "-INFINITY";
×
358
    else if (x.is_positive_infinity())
×
359
        str_ = "INFINITY";
×
360
};
×
361

362
void CPPSymbolicPrinter::bvisit(const SymEngine::BooleanAtom& x) { str_ = x.get_val() ? "true" : "false"; };
×
363

364
void CPPSymbolicPrinter::bvisit(const SymEngine::Symbol& x) {
27✔
365
    if (symbolic::is_nullptr(symbolic::symbol(x.get_name()))) {
27✔
366
        str_ = "(reinterpret_cast<uintptr_t>(nullptr))";
×
367
        return;
×
368
    }
×
369
    std::string name = x.get_name();
27✔
370

371
    if (this->function_.is_external(name)) {
27✔
372
        name = "(reinterpret_cast<uintptr_t>(&" + this->external_prefix_ + name + "))";
1✔
373
    } else if (this->function_.exists(name) && this->function_.type(name).type_id() == types::TypeID::Pointer) {
26✔
374
        name = "(reinterpret_cast<uintptr_t>(" + name + "))";
×
375
    }
×
376

377
    str_ = name;
27✔
378
};
27✔
379

380
void CPPSymbolicPrinter::bvisit(const SymEngine::And& x) {
×
381
    std::ostringstream s;
×
382
    auto container = x.get_container();
×
383
    s << apply(*container.begin());
×
384
    for (auto it = ++(container.begin()); it != container.end(); ++it) {
×
385
        s << " && " << apply(*it);
×
386
    }
×
387
    str_ = parenthesize(s.str());
×
388
};
×
389

390
void CPPSymbolicPrinter::bvisit(const SymEngine::Or& x) {
×
391
    std::ostringstream s;
×
392
    auto container = x.get_container();
×
393
    s << apply(*container.begin());
×
394
    for (auto it = ++(container.begin()); it != container.end(); ++it) {
×
395
        s << " || " << apply(*it);
×
396
    }
×
397
    str_ = parenthesize(s.str());
×
398
};
×
399

400
void CPPSymbolicPrinter::bvisit(const SymEngine::Not& x) {
×
401
    str_ = "!" + apply(x.get_arg());
×
402
    str_ = parenthesize(str_);
×
403
};
×
404

405
void CPPSymbolicPrinter::bvisit(const SymEngine::Equality& x) {
×
406
    str_ = apply(x.get_args()[0]) + " == " + apply(x.get_args()[1]);
×
407
    str_ = parenthesize(str_);
×
408
};
×
409

410
void CPPSymbolicPrinter::bvisit(const SymEngine::Unequality& x) {
×
411
    str_ = apply(x.get_args()[0]) + " != " + apply(x.get_args()[1]);
×
412
    str_ = parenthesize(str_);
×
413
};
×
414

415
void CPPSymbolicPrinter::bvisit(const SymEngine::Min& x) {
×
416
    std::ostringstream s;
×
417
    auto container = x.get_args();
×
418
    if (container.size() == 1) {
×
419
        s << apply(*container.begin());
×
420
    } else {
×
421
        s << "__daisy_min(";
×
422
        s << apply(*container.begin());
×
423

424
        // Recursively apply __daisy_min to the arguments
425
        SymEngine::vec_basic subargs;
×
426
        for (auto it = ++(container.begin()); it != container.end(); ++it) {
×
427
            subargs.push_back(*it);
×
428
        }
×
429
        auto submin = SymEngine::min(subargs);
×
430
        s << ", " << apply(submin);
×
431

432
        s << ")";
×
433
    }
×
434

435
    str_ = s.str();
×
436
};
×
437

438
void CPPSymbolicPrinter::bvisit(const SymEngine::Max& x) {
×
439
    std::ostringstream s;
×
440
    auto container = x.get_args();
×
441
    if (container.size() == 1) {
×
442
        s << apply(*container.begin());
×
443
    } else {
×
444
        s << "__daisy_max(";
×
445
        s << apply(*container.begin());
×
446

447
        // Recursively apply __daisy_max to the arguments
448
        SymEngine::vec_basic subargs;
×
449
        for (auto it = ++(container.begin()); it != container.end(); ++it) {
×
450
            subargs.push_back(*it);
×
451
        }
×
452
        auto submax = SymEngine::max(subargs);
×
453
        s << ", " << apply(submax);
×
454

455
        s << ")";
×
456
    }
×
457

458
    str_ = s.str();
×
459
};
×
460

461
void CPPSymbolicPrinter::bvisit(const SymEngine::FunctionSymbol& x) {
1✔
462
    if (x.get_name() == "idiv") {
1✔
463
        str_ = "((" + apply(x.get_args()[0]) + ") / (" + apply(x.get_args()[1]) + "))";
×
464
    } else if (x.get_name() == "iabs") {
1✔
465
        str_ = "((" + apply(x.get_args()[0]) + ") < 0 ? -(" + apply(x.get_args()[0]) + ") : (" +
×
466
               apply(x.get_args()[0]) + "))";
×
467
    } else if (x.get_name() == "zext_i64") {
1✔
468
        str_ = "((long long) ((unsigned long long) (" + apply(x.get_args()[0]) + ")))";
×
469
    } else if (x.get_name() == "trunc_i32") {
1✔
470
        str_ = "((int) ((unsigned int) ((unsigned long long) (" + apply(x.get_args()[0]) + "))))";
×
471
    } else if (x.get_name() == "imod") {
1✔
NEW
472
        str_ = "((" + apply(x.get_args()[0]) + ") % (" + apply(x.get_args()[1]) + "))";
×
473
    } else if (x.get_name() == "sizeof") {
1✔
474
        auto& so = dynamic_cast<const symbolic::SizeOfTypeFunction&>(x);
1✔
475
        auto& type = so.get_type();
1✔
476
        CPPLanguageExtension lang(this->function_, this->external_prefix_);
1✔
477
        str_ = "sizeof(" + lang.declaration("", type) + ")";
1✔
478
    } else if (x.get_name() == "malloc_usable_size") {
1✔
479
        str_ = "malloc_usable_size(" +
×
480
               SymEngine::rcp_static_cast<const SymEngine::Symbol>(x.get_args()[0])->get_name() + ")";
×
481
    } else {
×
482
        throw std::runtime_error("Unsupported function symbol: " + x.get_name());
×
483
    }
×
484
};
1✔
485

486
void CPPSymbolicPrinter::_print_pow(
487
    std::ostringstream& o,
488
    const SymEngine::RCP<const SymEngine::Basic>& a,
489
    const SymEngine::RCP<const SymEngine::Basic>& b
490
) {
2✔
491
    if (SymEngine::eq(*a, *SymEngine::E)) {
2✔
492
        o << "exp(" << apply(b) << ")";
×
493
    } else if (SymEngine::eq(*b, *SymEngine::rational(1, 2))) {
2✔
494
        o << "sqrt(" << apply(a) << ")";
×
495
    } else if (SymEngine::eq(*b, *SymEngine::rational(1, 3))) {
2✔
496
        o << "cbrt(" << apply(a) << ")";
×
497
    } else if (SymEngine::eq(*b, *SymEngine::integer(2))) {
2✔
498
        o << "((" + apply(a) + ") * (" + apply(a) + "))";
2✔
499
    } else {
2✔
500
        o << "__daisy_sym_pow(" << apply(a) << ", " << apply(b) << ")";
×
501
    }
×
502
};
2✔
503

504
} // namespace codegen
505
} // 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