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

daisytuner / docc / 23249128422

18 Mar 2026 02:12PM UTC coverage: 63.938% (+0.3%) from 63.617%
23249128422

Pull #584

github

web-flow
Merge 0fcde60dc into 64d54d7de
Pull Request #584: adds diamond tiling test

18 of 20 new or added lines in 1 file covered. (90.0%)

1180 existing lines in 28 files now uncovered.

26122 of 40855 relevant lines covered (63.94%)

407.69 hits per line

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

75.82
/sdfg/src/serializer/json_serializer.cpp
1
#include "sdfg/serializer/json_serializer.h"
2

3
#include <cassert>
4
#include <memory>
5
#include <sdfg/data_flow/library_nodes/load_const_node.h>
6
#include <utility>
7
#include <vector>
8

9
#include "sdfg/data_flow/library_nodes/barrier_local_node.h"
10
#include "sdfg/data_flow/library_nodes/call_node.h"
11
#include "sdfg/data_flow/library_nodes/invoke_node.h"
12
#include "sdfg/data_flow/library_nodes/math/math.h"
13
#include "sdfg/data_flow/library_nodes/math/tensor/elementwise_ops/cmath_node.h"
14
#include "sdfg/data_flow/library_nodes/math/tensor/elementwise_ops/tasklet_node.h"
15
#include "sdfg/data_flow/library_nodes/metadata_node.h"
16
#include "sdfg/data_flow/library_nodes/stdlib/stdlib.h"
17

18
#include "sdfg/analysis/users.h"
19
#include "sdfg/builder/structured_sdfg_builder.h"
20
#include "sdfg/data_flow/library_node.h"
21
#include "sdfg/element.h"
22
#include "sdfg/structured_control_flow/block.h"
23
#include "sdfg/structured_control_flow/for.h"
24
#include "sdfg/structured_control_flow/if_else.h"
25
#include "sdfg/structured_control_flow/map.h"
26
#include "sdfg/structured_control_flow/return.h"
27
#include "sdfg/structured_control_flow/sequence.h"
28
#include "sdfg/structured_control_flow/while.h"
29
#include "sdfg/structured_sdfg.h"
30
#include "sdfg/symbolic/symbolic.h"
31
#include "sdfg/types/function.h"
32
#include "sdfg/types/scalar.h"
33
#include "sdfg/types/type.h"
34
#include "symengine/expression.h"
35
#include "symengine/logic.h"
36
#include "symengine/symengine_rcp.h"
37

38
namespace sdfg {
39
namespace serializer {
40

41
FunctionType function_type_from_string(const std::string& str) {
27✔
42
    if (str == FunctionType_CPU.value()) {
27✔
43
        return FunctionType_CPU;
27✔
44
    } else if (str == FunctionType_NV_GLOBAL.value()) {
27✔
45
        return FunctionType_NV_GLOBAL;
×
46
    }
×
47

48
    return FunctionType(str);
×
49
}
27✔
50

51
/*
52
 * * JSONSerializer class
53
 * * Serialization logic
54
 */
55

56
nlohmann::json JSONSerializer::serialize(
57
    const sdfg::StructuredSDFG& sdfg,
58
    analysis::AnalysisManager* analysis_manager,
59
    structured_control_flow::Sequence* root
60
) {
26✔
61
    nlohmann::json j;
26✔
62

63
    j["name"] = sdfg.name();
26✔
64
    j["element_counter"] = sdfg.element_counter();
26✔
65
    j["type"] = std::string(sdfg.type().value());
26✔
66

67
    nlohmann::json return_type_json;
26✔
68
    type_to_json(return_type_json, sdfg.return_type());
26✔
69
    j["return_type"] = return_type_json;
26✔
70

71
    j["structures"] = nlohmann::json::array();
26✔
72
    for (const auto& structure_name : sdfg.structures()) {
26✔
73
        const auto& structure = sdfg.structure(structure_name);
1✔
74
        nlohmann::json structure_json;
1✔
75
        structure_definition_to_json(structure_json, structure);
1✔
76
        j["structures"].push_back(structure_json);
1✔
77
    }
1✔
78

79
    j["containers"] = nlohmann::json::object();
26✔
80
    for (const auto& container : sdfg.containers()) {
147✔
81
        nlohmann::json desc;
147✔
82
        type_to_json(desc, sdfg.type(container));
147✔
83
        j["containers"][container] = desc;
147✔
84
    }
147✔
85

86
    j["arguments"] = nlohmann::json::array();
26✔
87
    for (const auto& argument : sdfg.arguments()) {
30✔
88
        j["arguments"].push_back(argument);
30✔
89
    }
30✔
90

91
    j["externals"] = nlohmann::json::array();
26✔
92
    for (const auto& external : sdfg.externals()) {
26✔
93
        nlohmann::json external_json;
7✔
94
        external_json["name"] = external;
7✔
95
        external_json["linkage_type"] = sdfg.linkage_type(external);
7✔
96
        j["externals"].push_back(external_json);
7✔
97
    }
7✔
98

99
    j["metadata"] = nlohmann::json::object();
26✔
100
    for (const auto& entry : sdfg.metadata()) {
26✔
101
        j["metadata"][entry.first] = entry.second;
1✔
102
    }
1✔
103

104
    // Walk the SDFG
105
    nlohmann::json root_json;
26✔
106
    sequence_to_json(root_json, sdfg.root());
26✔
107
    j["root"] = root_json;
26✔
108

109
    return j;
26✔
110
}
26✔
111

112
void JSONSerializer::dataflow_to_json(nlohmann::json& j, const data_flow::DataFlowGraph& dataflow) {
58✔
113
    j["type"] = "dataflow";
58✔
114
    j["nodes"] = nlohmann::json::array();
58✔
115
    j["edges"] = nlohmann::json::array();
58✔
116

117
    for (auto& node : dataflow.nodes()) {
219✔
118
        nlohmann::json node_json;
219✔
119
        node_json["element_id"] = node.element_id();
219✔
120

121
        node_json["debug_info"] = nlohmann::json::object();
219✔
122
        debug_info_to_json(node_json["debug_info"], node.debug_info());
219✔
123

124
        if (auto tasklet = dynamic_cast<const data_flow::Tasklet*>(&node)) {
219✔
125
            node_json["type"] = "tasklet";
61✔
126
            node_json["code"] = tasklet->code();
61✔
127
            node_json["inputs"] = nlohmann::json::array();
61✔
128
            for (auto& input : tasklet->inputs()) {
101✔
129
                node_json["inputs"].push_back(input);
101✔
130
            }
101✔
131
            node_json["output"] = tasklet->output();
61✔
132
        } else if (auto lib_node = dynamic_cast<const data_flow::LibraryNode*>(&node)) {
158✔
133
            node_json["type"] = "library_node";
12✔
134
            node_json["implementation_type"] = std::string(lib_node->implementation_type().value());
12✔
135
            auto serializer_fn =
12✔
136
                LibraryNodeSerializerRegistry::instance().get_library_node_serializer(lib_node->code().value());
12✔
137
            if (serializer_fn == nullptr) {
12✔
138
                throw std::runtime_error("Unknown library node code: " + std::string(lib_node->code().value()));
×
139
            }
×
140
            auto serializer = serializer_fn();
12✔
141
            auto lib_node_json = serializer->serialize(*lib_node);
12✔
142
            node_json.merge_patch(lib_node_json);
12✔
143
        } else if (auto code_node = dynamic_cast<const data_flow::ConstantNode*>(&node)) {
146✔
144
            node_json["type"] = "constant_node";
×
145
            node_json["data"] = code_node->data();
×
146

147
            nlohmann::json type_json;
×
148
            type_to_json(type_json, code_node->type());
×
149
            node_json["data_type"] = type_json;
×
150
        } else if (auto code_node = dynamic_cast<const data_flow::AccessNode*>(&node)) {
146✔
151
            node_json["type"] = "access_node";
146✔
152
            node_json["data"] = code_node->data();
146✔
153
        } else {
146✔
154
            throw std::runtime_error("Unknown node type");
×
155
        }
×
156

157
        j["nodes"].push_back(node_json);
219✔
158
    }
219✔
159

160
    for (auto& edge : dataflow.edges()) {
191✔
161
        nlohmann::json edge_json;
191✔
162
        edge_json["element_id"] = edge.element_id();
191✔
163

164
        edge_json["debug_info"] = nlohmann::json::object();
191✔
165
        debug_info_to_json(edge_json["debug_info"], edge.debug_info());
191✔
166

167
        edge_json["src"] = edge.src().element_id();
191✔
168
        edge_json["dst"] = edge.dst().element_id();
191✔
169

170
        edge_json["src_conn"] = edge.src_conn();
191✔
171
        edge_json["dst_conn"] = edge.dst_conn();
191✔
172

173
        edge_json["subset"] = nlohmann::json::array();
191✔
174
        for (auto& subset : edge.subset()) {
191✔
175
            edge_json["subset"].push_back(expression(subset));
72✔
176
        }
72✔
177

178
        nlohmann::json base_type_json;
191✔
179
        type_to_json(base_type_json, edge.base_type());
191✔
180
        edge_json["base_type"] = base_type_json;
191✔
181

182
        j["edges"].push_back(edge_json);
191✔
183
    }
191✔
184
}
58✔
185

186
void JSONSerializer::block_to_json(nlohmann::json& j, const structured_control_flow::Block& block) {
55✔
187
    j["type"] = "block";
55✔
188
    j["element_id"] = block.element_id();
55✔
189

190
    j["debug_info"] = nlohmann::json::object();
55✔
191
    debug_info_to_json(j["debug_info"], block.debug_info());
55✔
192

193
    nlohmann::json dataflow_json;
55✔
194
    dataflow_to_json(dataflow_json, block.dataflow());
55✔
195
    j["dataflow"] = dataflow_json;
55✔
196
}
55✔
197

198
void JSONSerializer::for_to_json(nlohmann::json& j, const structured_control_flow::For& for_node) {
18✔
199
    j["type"] = "for";
18✔
200
    j["element_id"] = for_node.element_id();
18✔
201

202
    j["debug_info"] = nlohmann::json::object();
18✔
203
    debug_info_to_json(j["debug_info"], for_node.debug_info());
18✔
204

205
    j["indvar"] = expression(for_node.indvar());
18✔
206
    j["init"] = expression(for_node.init());
18✔
207
    j["condition"] = expression(for_node.condition());
18✔
208
    j["update"] = expression(for_node.update());
18✔
209

210
    nlohmann::json body_json;
18✔
211
    sequence_to_json(body_json, for_node.root());
18✔
212
    j["root"] = body_json;
18✔
213
}
18✔
214

215
void JSONSerializer::if_else_to_json(nlohmann::json& j, const structured_control_flow::IfElse& if_else_node) {
2✔
216
    j["type"] = "if_else";
2✔
217
    j["element_id"] = if_else_node.element_id();
2✔
218

219
    j["debug_info"] = nlohmann::json::object();
2✔
220
    debug_info_to_json(j["debug_info"], if_else_node.debug_info());
2✔
221

222
    j["branches"] = nlohmann::json::array();
2✔
223
    for (size_t i = 0; i < if_else_node.size(); i++) {
6✔
224
        nlohmann::json branch_json;
4✔
225
        branch_json["condition"] = expression(if_else_node.at(i).second);
4✔
226
        nlohmann::json body_json;
4✔
227
        sequence_to_json(body_json, if_else_node.at(i).first);
4✔
228
        branch_json["root"] = body_json;
4✔
229
        j["branches"].push_back(branch_json);
4✔
230
    }
4✔
231
}
2✔
232

233
void JSONSerializer::while_node_to_json(nlohmann::json& j, const structured_control_flow::While& while_node) {
6✔
234
    j["type"] = "while";
6✔
235
    j["element_id"] = while_node.element_id();
6✔
236

237
    j["debug_info"] = nlohmann::json::object();
6✔
238
    debug_info_to_json(j["debug_info"], while_node.debug_info());
6✔
239

240
    nlohmann::json body_json;
6✔
241
    sequence_to_json(body_json, while_node.root());
6✔
242
    j["root"] = body_json;
6✔
243
}
6✔
244

245
void JSONSerializer::break_node_to_json(nlohmann::json& j, const structured_control_flow::Break& break_node) {
2✔
246
    j["type"] = "break";
2✔
247
    j["element_id"] = break_node.element_id();
2✔
248

249
    j["debug_info"] = nlohmann::json::object();
2✔
250
    debug_info_to_json(j["debug_info"], break_node.debug_info());
2✔
251
}
2✔
252

253
void JSONSerializer::continue_node_to_json(nlohmann::json& j, const structured_control_flow::Continue& continue_node) {
2✔
254
    j["type"] = "continue";
2✔
255
    j["element_id"] = continue_node.element_id();
2✔
256

257
    j["debug_info"] = nlohmann::json::object();
2✔
258
    debug_info_to_json(j["debug_info"], continue_node.debug_info());
2✔
259
}
2✔
260

261
void JSONSerializer::map_to_json(nlohmann::json& j, const structured_control_flow::Map& map_node) {
10✔
262
    j["type"] = "map";
10✔
263
    j["element_id"] = map_node.element_id();
10✔
264

265
    j["debug_info"] = nlohmann::json::object();
10✔
266
    debug_info_to_json(j["debug_info"], map_node.debug_info());
10✔
267

268
    j["indvar"] = expression(map_node.indvar());
10✔
269
    j["init"] = expression(map_node.init());
10✔
270
    j["condition"] = expression(map_node.condition());
10✔
271
    j["update"] = expression(map_node.update());
10✔
272

273
    j["schedule_type"] = nlohmann::json::object();
10✔
274
    schedule_type_to_json(j["schedule_type"], map_node.schedule_type());
10✔
275

276
    nlohmann::json body_json;
10✔
277
    sequence_to_json(body_json, map_node.root());
10✔
278
    j["root"] = body_json;
10✔
279
}
10✔
280

281
void JSONSerializer::return_node_to_json(nlohmann::json& j, const structured_control_flow::Return& return_node) {
2✔
282
    j["type"] = "return";
2✔
283
    j["element_id"] = return_node.element_id();
2✔
284
    j["data"] = return_node.data();
2✔
285

286
    if (return_node.is_constant()) {
2✔
287
        nlohmann::json type_json;
×
288
        type_to_json(type_json, return_node.type());
×
289
        j["data_type"] = type_json;
×
290
    }
×
291

292
    j["debug_info"] = nlohmann::json::object();
2✔
293
    debug_info_to_json(j["debug_info"], return_node.debug_info());
2✔
294
}
2✔
295

296
void JSONSerializer::sequence_to_json(nlohmann::json& j, const structured_control_flow::Sequence& sequence) {
68✔
297
    j["type"] = "sequence";
68✔
298
    j["element_id"] = sequence.element_id();
68✔
299

300
    j["debug_info"] = nlohmann::json::object();
68✔
301
    debug_info_to_json(j["debug_info"], sequence.debug_info());
68✔
302

303
    j["children"] = nlohmann::json::array();
68✔
304
    j["transitions"] = nlohmann::json::array();
68✔
305

306
    for (size_t i = 0; i < sequence.size(); i++) {
150✔
307
        nlohmann::json child_json;
82✔
308
        auto& child = sequence.at(i).first;
82✔
309
        auto& transition = sequence.at(i).second;
82✔
310

311
        if (auto block = dynamic_cast<const structured_control_flow::Block*>(&child)) {
82✔
312
            block_to_json(child_json, *block);
53✔
313
        } else if (auto for_node = dynamic_cast<const structured_control_flow::For*>(&child)) {
53✔
314
            for_to_json(child_json, *for_node);
16✔
315
        } else if (auto sequence_node = dynamic_cast<const structured_control_flow::Sequence*>(&child)) {
16✔
316
            sequence_to_json(child_json, *sequence_node);
×
317
        } else if (auto condition_node = dynamic_cast<const structured_control_flow::IfElse*>(&child)) {
13✔
318
            if_else_to_json(child_json, *condition_node);
×
319
        } else if (auto while_node = dynamic_cast<const structured_control_flow::While*>(&child)) {
13✔
320
            while_node_to_json(child_json, *while_node);
1✔
321
        } else if (auto return_node = dynamic_cast<const structured_control_flow::Return*>(&child)) {
12✔
322
            return_node_to_json(child_json, *return_node);
×
323
        } else if (auto break_node = dynamic_cast<const structured_control_flow::Break*>(&child)) {
12✔
324
            break_node_to_json(child_json, *break_node);
2✔
325
        } else if (auto continue_node = dynamic_cast<const structured_control_flow::Continue*>(&child)) {
10✔
326
            continue_node_to_json(child_json, *continue_node);
2✔
327
        } else if (auto map_node = dynamic_cast<const structured_control_flow::Map*>(&child)) {
8✔
328
            map_to_json(child_json, *map_node);
8✔
329
        } else {
8✔
330
            throw std::runtime_error("Unknown child type");
×
331
        }
×
332

333
        j["children"].push_back(child_json);
82✔
334

335
        // Add transition information
336
        nlohmann::json transition_json;
82✔
337
        transition_json["type"] = "transition";
82✔
338
        transition_json["element_id"] = transition.element_id();
82✔
339

340
        transition_json["debug_info"] = nlohmann::json::object();
82✔
341
        debug_info_to_json(transition_json["debug_info"], transition.debug_info());
82✔
342

343
        transition_json["assignments"] = nlohmann::json::array();
82✔
344
        for (const auto& assignment : transition.assignments()) {
82✔
345
            nlohmann::json assignment_json;
3✔
346
            assignment_json["symbol"] = expression(assignment.first);
3✔
347
            assignment_json["expression"] = expression(assignment.second);
3✔
348
            transition_json["assignments"].push_back(assignment_json);
3✔
349
        }
3✔
350

351
        j["transitions"].push_back(transition_json);
82✔
352
    }
82✔
353
}
68✔
354

355
void JSONSerializer::type_to_json(nlohmann::json& j, const types::IType& type) {
564✔
356
    if (auto scalar_type = dynamic_cast<const types::Scalar*>(&type)) {
564✔
357
        j["type"] = "scalar";
403✔
358
        j["primitive_type"] = scalar_type->primitive_type();
403✔
359
        j["storage_type"] = nlohmann::json::object();
403✔
360
        storage_type_to_json(j["storage_type"], scalar_type->storage_type());
403✔
361
        j["initializer"] = scalar_type->initializer();
403✔
362
        j["alignment"] = scalar_type->alignment();
403✔
363
    } else if (auto array_type = dynamic_cast<const types::Array*>(&type)) {
403✔
364
        j["type"] = "array";
76✔
365
        nlohmann::json element_type_json;
76✔
366
        type_to_json(element_type_json, array_type->element_type());
76✔
367
        j["element_type"] = element_type_json;
76✔
368
        j["num_elements"] = expression(array_type->num_elements());
76✔
369
        j["storage_type"] = nlohmann::json::object();
76✔
370
        storage_type_to_json(j["storage_type"], array_type->storage_type());
76✔
371
        j["initializer"] = array_type->initializer();
76✔
372
        j["alignment"] = array_type->alignment();
76✔
373
    } else if (auto pointer_type = dynamic_cast<const types::Pointer*>(&type)) {
85✔
374
        j["type"] = "pointer";
67✔
375
        if (pointer_type->has_pointee_type()) {
67✔
376
            nlohmann::json pointee_type_json;
54✔
377
            type_to_json(pointee_type_json, pointer_type->pointee_type());
54✔
378
            j["pointee_type"] = pointee_type_json;
54✔
379
        }
54✔
380
        j["storage_type"] = nlohmann::json::object();
67✔
381
        storage_type_to_json(j["storage_type"], pointer_type->storage_type());
67✔
382
        j["initializer"] = pointer_type->initializer();
67✔
383
        j["alignment"] = pointer_type->alignment();
67✔
384
    } else if (auto structure_type = dynamic_cast<const types::Structure*>(&type)) {
67✔
385
        j["type"] = "structure";
5✔
386
        j["name"] = structure_type->name();
5✔
387
        j["storage_type"] = nlohmann::json::object();
5✔
388
        storage_type_to_json(j["storage_type"], structure_type->storage_type());
5✔
389
        j["initializer"] = structure_type->initializer();
5✔
390
        j["alignment"] = structure_type->alignment();
5✔
391
    } else if (auto function_type = dynamic_cast<const types::Function*>(&type)) {
13✔
392
        j["type"] = "function";
5✔
393
        nlohmann::json return_type_json;
5✔
394
        type_to_json(return_type_json, function_type->return_type());
5✔
395
        j["return_type"] = return_type_json;
5✔
396
        j["params"] = nlohmann::json::array();
5✔
397
        for (size_t i = 0; i < function_type->num_params(); i++) {
16✔
398
            nlohmann::json param_json;
11✔
399
            type_to_json(param_json, function_type->param_type(symbolic::integer(i)));
11✔
400
            j["params"].push_back(param_json);
11✔
401
        }
11✔
402
        j["is_var_arg"] = function_type->is_var_arg();
5✔
403
        j["storage_type"] = nlohmann::json::object();
5✔
404
        storage_type_to_json(j["storage_type"], function_type->storage_type());
5✔
405
        j["initializer"] = function_type->initializer();
5✔
406
        j["alignment"] = function_type->alignment();
5✔
407
    } else if (auto reference_type = dynamic_cast<const sdfg::codegen::Reference*>(&type)) {
8✔
408
        j["type"] = "reference";
8✔
409
        nlohmann::json reference_type_json;
8✔
410
        type_to_json(reference_type_json, reference_type->reference_type());
8✔
411
        j["reference_type"] = reference_type_json;
8✔
412
    } else if (auto tensor_type = dynamic_cast<const types::Tensor*>(&type)) {
8✔
413
        j["type"] = "tensor";
×
414
        nlohmann::json element_type_json;
×
415
        type_to_json(element_type_json, tensor_type->element_type());
×
416
        j["element_type"] = element_type_json;
×
417
        j["shape"] = nlohmann::json::array();
×
418
        for (const auto& dim : tensor_type->shape()) {
×
419
            j["shape"].push_back(expression(dim));
×
420
        }
×
421
        j["strides"] = nlohmann::json::array();
×
422
        for (const auto& stride : tensor_type->strides()) {
×
423
            j["strides"].push_back(expression(stride));
×
424
        }
×
425
        j["offset"] = expression(tensor_type->offset());
×
426
        j["storage_type"] = nlohmann::json::object();
×
427
        storage_type_to_json(j["storage_type"], tensor_type->storage_type());
×
428
        j["initializer"] = tensor_type->initializer();
×
429
        j["alignment"] = tensor_type->alignment();
×
430
    } else {
×
431
        throw std::runtime_error("Unknown type");
×
432
    }
×
433
}
564✔
434

435
void JSONSerializer::structure_definition_to_json(nlohmann::json& j, const types::StructureDefinition& definition) {
2✔
436
    j["name"] = definition.name();
2✔
437
    j["members"] = nlohmann::json::array();
2✔
438
    for (size_t i = 0; i < definition.num_members(); i++) {
4✔
439
        nlohmann::json member_json;
2✔
440
        type_to_json(member_json, definition.member_type(symbolic::integer(i)));
2✔
441
        j["members"].push_back(member_json);
2✔
442
    }
2✔
443
    j["is_packed"] = definition.is_packed();
2✔
444
}
2✔
445

446
void JSONSerializer::debug_info_to_json(nlohmann::json& j, const DebugInfo& debug_info) {
657✔
447
    j["has"] = debug_info.has();
657✔
448
    j["filename"] = debug_info.filename();
657✔
449
    j["function"] = debug_info.function();
657✔
450
    j["start_line"] = debug_info.start_line();
657✔
451
    j["start_column"] = debug_info.start_column();
657✔
452
    j["end_line"] = debug_info.end_line();
657✔
453
    j["end_column"] = debug_info.end_column();
657✔
454
}
657✔
455

456

457
void JSONSerializer::schedule_type_to_json(nlohmann::json& j, const ScheduleType& schedule_type) {
12✔
458
    j["value"] = schedule_type.value();
12✔
459
    j["category"] = static_cast<int>(schedule_type.category());
12✔
460
    j["properties"] = nlohmann::json::object();
12✔
461
    for (const auto& prop : schedule_type.properties()) {
12✔
462
        j["properties"][prop.first] = prop.second;
12✔
463
    }
12✔
464
}
12✔
465

466
void JSONSerializer::storage_type_to_json(nlohmann::json& j, const types::StorageType& storage_type) {
556✔
467
    j["value"] = storage_type.value();
556✔
468
    j["allocation"] = storage_type.allocation();
556✔
469
    j["deallocation"] = storage_type.deallocation();
556✔
470
    if (!storage_type.allocation_size().is_null()) {
556✔
471
        j["allocation_size"] = expression(storage_type.allocation_size());
6✔
472
    }
6✔
473
    const symbolic::Expression& arg1 = storage_type.arg1();
556✔
474
    if (!arg1.is_null()) {
556✔
475
        auto args = nlohmann::json::array();
×
476
        args.push_back(expression(arg1));
×
477
        j["args"] = args;
×
478
    }
×
479
}
556✔
480

481

482
/*
483
 * * Deserialization logic
484
 */
485

486
std::unique_ptr<StructuredSDFG> JSONSerializer::deserialize(nlohmann::json& j) {
27✔
487
    assert(j.contains("name"));
27✔
488
    assert(j["name"].is_string());
27✔
489
    assert(j.contains("type"));
27✔
490
    assert(j["type"].is_string());
27✔
491
    assert(j.contains("element_counter"));
27✔
492
    assert(j["element_counter"].is_number_integer());
27✔
493

494
    std::unique_ptr<types::IType> return_type;
27✔
495
    if (j.contains("return_type")) {
27✔
496
        return_type = json_to_type(j["return_type"]);
27✔
497
    } else {
27✔
498
        return_type = std::make_unique<types::Scalar>(types::PrimitiveType::Void);
×
499
    }
×
500

501
    FunctionType function_type = function_type_from_string(j["type"].get<std::string>());
27✔
502
    builder::StructuredSDFGBuilder builder(j["name"], function_type, *return_type);
27✔
503

504
    size_t element_counter = j["element_counter"];
27✔
505
    builder.set_element_counter(element_counter);
27✔
506

507
    // deserialize structures
508
    assert(j.contains("structures"));
27✔
509
    assert(j["structures"].is_array());
27✔
510
    for (const auto& structure : j["structures"]) {
27✔
511
        assert(structure.contains("name"));
1✔
512
        assert(structure["name"].is_string());
1✔
513
        json_to_structure_definition(structure, builder);
1✔
514
    }
1✔
515

516
    nlohmann::json& containers = j["containers"];
27✔
517

518
    // deserialize externals
519
    for (const auto& external : j["externals"]) {
27✔
520
        assert(external.contains("name"));
7✔
521
        assert(external["name"].is_string());
7✔
522
        assert(external.contains("linkage_type"));
7✔
523
        assert(external["linkage_type"].is_number_integer());
7✔
524
        auto& type_desc = containers.at(external["name"].get<std::string>());
7✔
525
        auto type = json_to_type(type_desc);
7✔
526
        builder.add_external(external["name"], *type, LinkageType(external["linkage_type"]));
7✔
527
    }
7✔
528

529
    // deserialize arguments
530
    for (const auto& name : j["arguments"]) {
33✔
531
        auto& type_desc = containers.at(name.get<std::string>());
33✔
532
        auto type = json_to_type(type_desc);
33✔
533
        builder.add_container(name, *type, true, false);
33✔
534
    }
33✔
535

536
    // deserialize transients
537
    for (const auto& entry : containers.items()) {
153✔
538
        if (builder.subject().is_argument(entry.key())) {
153✔
539
            continue;
33✔
540
        }
33✔
541
        if (builder.subject().is_external(entry.key())) {
120✔
542
            continue;
7✔
543
        }
7✔
544
        auto type = json_to_type(entry.value());
113✔
545
        builder.add_container(entry.key(), *type, false, false);
113✔
546
    }
113✔
547

548
    // deserialize root node
549
    assert(j.contains("root"));
27✔
550
    auto& root = builder.subject().root();
27✔
551
    json_to_sequence(j["root"], builder, root);
27✔
552

553
    // deserialize metadata
554
    assert(j.contains("metadata"));
27✔
555
    assert(j["metadata"].is_object());
27✔
556
    for (const auto& entry : j["metadata"].items()) {
27✔
557
        builder.subject().add_metadata(entry.key(), entry.value());
1✔
558
    }
1✔
559

560
    builder.set_element_counter(element_counter);
27✔
561

562
    return builder.move();
27✔
563
}
27✔
564

565
void JSONSerializer::json_to_structure_definition(const nlohmann::json& j, builder::StructuredSDFGBuilder& builder) {
2✔
566
    assert(j.contains("name"));
2✔
567
    assert(j["name"].is_string());
2✔
568
    assert(j.contains("members"));
2✔
569
    assert(j["members"].is_array());
2✔
570
    assert(j.contains("is_packed"));
2✔
571
    assert(j["is_packed"].is_boolean());
2✔
572
    auto is_packed = j["is_packed"];
2✔
573
    auto& definition = builder.add_structure(j["name"], is_packed);
2✔
574
    for (const auto& member : j["members"]) {
2✔
575
        nlohmann::json member_json;
2✔
576
        auto member_type = json_to_type(member);
2✔
577
        definition.add_member(*member_type);
2✔
578
    }
2✔
579
}
2✔
580

581
std::vector<std::pair<std::string, types::Scalar>> JSONSerializer::json_to_arguments(const nlohmann::json& j) {
×
582
    std::vector<std::pair<std::string, types::Scalar>> arguments;
×
583
    for (const auto& argument : j) {
×
584
        assert(argument.contains("name"));
×
585
        assert(argument["name"].is_string());
×
586
        assert(argument.contains("type"));
×
587
        assert(argument["type"].is_object());
×
588
        std::string name = argument["name"];
×
589
        auto type = json_to_type(argument["type"]);
×
590
        arguments.emplace_back(name, *dynamic_cast<types::Scalar*>(type.get()));
×
591
    }
×
592
    return arguments;
×
593
}
×
594

595
void JSONSerializer::json_to_dataflow(
596
    const nlohmann::json& j, builder::StructuredSDFGBuilder& builder, structured_control_flow::Block& parent
597
) {
48✔
598
    std::map<size_t, data_flow::DataFlowNode&> nodes_map;
48✔
599

600
    assert(j.contains("nodes"));
48✔
601
    assert(j["nodes"].is_array());
48✔
602
    for (const auto& node : j["nodes"]) {
215✔
603
        assert(node.contains("type"));
215✔
604
        assert(node["type"].is_string());
215✔
605
        assert(node.contains("element_id"));
215✔
606
        assert(node["element_id"].is_number_integer());
215✔
607
        std::string type = node["type"];
215✔
608
        if (type == "tasklet") {
215✔
609
            assert(node.contains("code"));
60✔
610
            assert(node["code"].is_number_integer());
60✔
611
            assert(node.contains("inputs"));
60✔
612
            assert(node["inputs"].is_array());
60✔
613
            assert(node.contains("output"));
60✔
614
            assert(node["output"].is_string());
60✔
615
            auto inputs = node["inputs"].get<std::vector<std::string>>();
60✔
616

617
            auto& tasklet =
60✔
618
                builder
60✔
619
                    .add_tasklet(parent, node["code"], node["output"], inputs, json_to_debug_info(node["debug_info"]));
60✔
620
            tasklet.element_id_ = node["element_id"];
60✔
621
            nodes_map.insert({node["element_id"], tasklet});
60✔
622
        } else if (type == "library_node") {
155✔
623
            assert(node.contains("code"));
12✔
624
            data_flow::LibraryNodeCode code(node["code"].get<std::string>());
12✔
625

626
            auto serializer_fn = LibraryNodeSerializerRegistry::instance().get_library_node_serializer(code.value());
12✔
627
            if (serializer_fn == nullptr) {
12✔
628
                throw std::runtime_error("Unknown library node code: " + std::string(code.value()));
×
629
            }
×
630
            auto serializer = serializer_fn();
12✔
631
            auto& lib_node = serializer->deserialize(node, builder, parent);
12✔
632
            lib_node.implementation_type() =
12✔
633
                data_flow::ImplementationType(node["implementation_type"].get<std::string>());
12✔
634
            lib_node.element_id_ = node["element_id"];
12✔
635
            nodes_map.insert({node["element_id"], lib_node});
12✔
636
        } else if (type == "access_node") {
143✔
637
            assert(node.contains("data"));
143✔
638
            auto& access_node = builder.add_access(parent, node["data"], json_to_debug_info(node["debug_info"]));
143✔
639
            access_node.element_id_ = node["element_id"];
143✔
640
            nodes_map.insert({node["element_id"], access_node});
143✔
641
        } else if (type == "constant_node") {
143✔
642
            assert(node.contains("data"));
×
643
            assert(node.contains("data_type"));
×
644

645
            auto type = json_to_type(node["data_type"]);
×
646

647
            auto& constant_node =
×
648
                builder.add_constant(parent, node["data"], *type, json_to_debug_info(node["debug_info"]));
×
649
            constant_node.element_id_ = node["element_id"];
×
650
            nodes_map.insert({node["element_id"], constant_node});
×
651
        } else {
×
652
            throw std::runtime_error("Unknown node type");
×
653
        }
×
654
    }
215✔
655

656
    assert(j.contains("edges"));
48✔
657
    assert(j["edges"].is_array());
48✔
658
    for (const auto& edge : j["edges"]) {
188✔
659
        assert(edge.contains("src"));
188✔
660
        assert(edge["src"].is_number_integer());
188✔
661
        assert(edge.contains("dst"));
188✔
662
        assert(edge["dst"].is_number_integer());
188✔
663
        assert(edge.contains("src_conn"));
188✔
664
        assert(edge["src_conn"].is_string());
188✔
665
        assert(edge.contains("dst_conn"));
188✔
666
        assert(edge["dst_conn"].is_string());
188✔
667
        assert(edge.contains("subset"));
188✔
668
        assert(edge["subset"].is_array());
188✔
669

670
        assert(nodes_map.find(edge["src"]) != nodes_map.end());
188✔
671
        assert(nodes_map.find(edge["dst"]) != nodes_map.end());
188✔
672
        auto& source = nodes_map.at(edge["src"]);
188✔
673
        auto& target = nodes_map.at(edge["dst"]);
188✔
674

675
        auto base_type = json_to_type(edge["base_type"]);
188✔
676

677
        assert(edge.contains("subset"));
188✔
678
        assert(edge["subset"].is_array());
188✔
679
        std::vector<symbolic::Expression> subset;
188✔
680
        for (const auto& subset_ : edge["subset"]) {
188✔
681
            assert(subset_.is_string());
70✔
682
            std::string subset_str = subset_;
70✔
683
            auto expr = symbolic::parse(subset_str);
70✔
684
            subset.push_back(expr);
70✔
685
        }
70✔
686
        auto& memlet = builder.add_memlet(
188✔
687
            parent,
188✔
688
            source,
188✔
689
            edge["src_conn"],
188✔
690
            target,
188✔
691
            edge["dst_conn"],
188✔
692
            subset,
188✔
693
            *base_type,
188✔
694
            json_to_debug_info(edge["debug_info"])
188✔
695
        );
188✔
696
        memlet.element_id_ = edge["element_id"];
188✔
697
    }
188✔
698
}
48✔
699

700
void JSONSerializer::json_to_sequence(
701
    const nlohmann::json& j, builder::StructuredSDFGBuilder& builder, structured_control_flow::Sequence& sequence
702
) {
61✔
703
    assert(j.contains("type"));
61✔
704
    assert(j["type"].is_string());
61✔
705
    assert(j.contains("children"));
61✔
706
    assert(j["children"].is_array());
61✔
707
    assert(j.contains("transitions"));
61✔
708
    assert(j["transitions"].is_array());
61✔
709
    assert(j["transitions"].size() == j["children"].size());
61✔
710

711
    sequence.element_id_ = j["element_id"];
61✔
712
    sequence.debug_info_ = json_to_debug_info(j["debug_info"]);
61✔
713

714
    std::string type = j["type"];
61✔
715
    if (type == "sequence") {
61✔
716
        for (size_t i = 0; i < j["children"].size(); i++) {
133✔
717
            auto& child = j["children"][i];
72✔
718
            auto& transition = j["transitions"][i];
72✔
719
            assert(child.contains("type"));
72✔
720
            assert(child["type"].is_string());
72✔
721

722
            assert(transition.contains("type"));
72✔
723
            assert(transition["type"].is_string());
72✔
724
            assert(transition.contains("assignments"));
72✔
725
            assert(transition["assignments"].is_array());
72✔
726
            control_flow::Assignments assignments;
72✔
727
            for (const auto& assignment : transition["assignments"]) {
72✔
728
                assert(assignment.contains("symbol"));
2✔
729
                assert(assignment["symbol"].is_string());
2✔
730
                assert(assignment.contains("expression"));
2✔
731
                assert(assignment["expression"].is_string());
2✔
732
                auto expr = symbolic::parse(assignment["expression"].get<std::string>());
2✔
733
                assignments.insert({symbolic::symbol(assignment["symbol"]), expr});
2✔
734
            }
2✔
735

736
            if (child["type"] == "block") {
72✔
737
                json_to_block_node(child, builder, sequence, assignments);
45✔
738
            } else if (child["type"] == "for") {
45✔
739
                json_to_for_node(child, builder, sequence, assignments);
16✔
740
            } else if (child["type"] == "if_else") {
16✔
741
                json_to_if_else_node(child, builder, sequence, assignments);
×
742
            } else if (child["type"] == "while") {
11✔
743
                json_to_while_node(child, builder, sequence, assignments);
1✔
744
            } else if (child["type"] == "break") {
10✔
745
                json_to_break_node(child, builder, sequence, assignments);
1✔
746
            } else if (child["type"] == "continue") {
9✔
747
                json_to_continue_node(child, builder, sequence, assignments);
1✔
748
            } else if (child["type"] == "return") {
8✔
749
                json_to_return_node(child, builder, sequence, assignments);
×
750
            } else if (child["type"] == "map") {
8✔
751
                json_to_map_node(child, builder, sequence, assignments);
8✔
752
            } else if (child["type"] == "sequence") {
8✔
753
                auto& subseq = builder.add_sequence(sequence, assignments, json_to_debug_info(child["debug_info"]));
×
754
                json_to_sequence(child, builder, subseq);
×
755
            } else {
×
756
                throw std::runtime_error("Unknown child type");
×
757
            }
×
758

759
            sequence.at(i).second.debug_info_ = json_to_debug_info(transition["debug_info"]);
72✔
760
            sequence.at(i).second.element_id_ = transition["element_id"];
72✔
761
        }
72✔
762
    } else {
61✔
763
        throw std::runtime_error("expected sequence type");
×
764
    }
×
765
}
61✔
766

767
void JSONSerializer::json_to_block_node(
768
    const nlohmann::json& j,
769
    builder::StructuredSDFGBuilder& builder,
770
    structured_control_flow::Sequence& parent,
771
    control_flow::Assignments& assignments
772
) {
46✔
773
    assert(j.contains("type"));
46✔
774
    assert(j["type"].is_string());
46✔
775
    assert(j.contains("dataflow"));
46✔
776
    assert(j["dataflow"].is_object());
46✔
777
    auto& block = builder.add_block(parent, assignments, json_to_debug_info(j["debug_info"]));
46✔
778
    block.element_id_ = j["element_id"];
46✔
779
    assert(j["dataflow"].contains("type"));
46✔
780
    assert(j["dataflow"]["type"].is_string());
46✔
781
    std::string type = j["dataflow"]["type"];
46✔
782
    if (type == "dataflow") {
46✔
783
        json_to_dataflow(j["dataflow"], builder, block);
46✔
784
    } else {
46✔
785
        throw std::runtime_error("Unknown dataflow type");
×
786
    }
×
787
}
46✔
788

789
void JSONSerializer::json_to_for_node(
790
    const nlohmann::json& j,
791
    builder::StructuredSDFGBuilder& builder,
792
    structured_control_flow::Sequence& parent,
793
    control_flow::Assignments& assignments
794
) {
17✔
795
    assert(j.contains("type"));
17✔
796
    assert(j["type"].is_string());
17✔
797
    assert(j.contains("indvar"));
17✔
798
    assert(j["indvar"].is_string());
17✔
799
    assert(j.contains("init"));
17✔
800
    assert(j["init"].is_string());
17✔
801
    assert(j.contains("condition"));
17✔
802
    assert(j["condition"].is_string());
17✔
803
    assert(j.contains("update"));
17✔
804
    assert(j["update"].is_string());
17✔
805
    assert(j.contains("root"));
17✔
806
    assert(j["root"].is_object());
17✔
807

808
    symbolic::Symbol indvar = symbolic::symbol(j["indvar"]);
17✔
809
    auto init = symbolic::parse(j["init"].get<std::string>());
17✔
810
    auto update = symbolic::parse(j["update"].get<std::string>());
17✔
811

812
    auto condition_expr = symbolic::parse(j["condition"].get<std::string>());
17✔
813
    symbolic::Condition condition = SymEngine::rcp_dynamic_cast<const SymEngine::Boolean>(condition_expr);
17✔
814
    if (condition.is_null()) {
17✔
815
        throw InvalidSDFGException("For loop condition is not a boolean expression");
×
816
    }
×
817

818
    auto& for_node =
17✔
819
        builder.add_for(parent, indvar, condition, init, update, assignments, json_to_debug_info(j["debug_info"]));
17✔
820
    for_node.element_id_ = j["element_id"];
17✔
821

822
    assert(j["root"].contains("type"));
17✔
823
    assert(j["root"]["type"].is_string());
17✔
824
    assert(j["root"]["type"] == "sequence");
17✔
825
    json_to_sequence(j["root"], builder, for_node.root());
17✔
826
}
17✔
827

828
void JSONSerializer::json_to_if_else_node(
829
    const nlohmann::json& j,
830
    builder::StructuredSDFGBuilder& builder,
831
    structured_control_flow::Sequence& parent,
832
    control_flow::Assignments& assignments
833
) {
1✔
834
    assert(j.contains("type"));
1✔
835
    assert(j["type"].is_string());
1✔
836
    assert(j["type"] == "if_else");
1✔
837
    assert(j.contains("branches"));
1✔
838
    assert(j["branches"].is_array());
1✔
839
    auto& if_else_node = builder.add_if_else(parent, assignments, json_to_debug_info(j["debug_info"]));
1✔
840
    if_else_node.element_id_ = j["element_id"];
1✔
841
    for (const auto& branch : j["branches"]) {
2✔
842
        assert(branch.contains("condition"));
2✔
843
        assert(branch["condition"].is_string());
2✔
844
        assert(branch.contains("root"));
2✔
845
        assert(branch["root"].is_object());
2✔
846

847
        auto condition_expr = symbolic::parse(branch["condition"].get<std::string>());
2✔
848
        symbolic::Condition condition = SymEngine::rcp_dynamic_cast<const SymEngine::Boolean>(condition_expr);
2✔
849
        if (condition.is_null()) {
2✔
850
            throw InvalidSDFGException("If condition is not a boolean expression");
×
851
        }
×
852
        auto& branch_node = builder.add_case(if_else_node, condition);
2✔
853
        assert(branch["root"].contains("type"));
2✔
854
        assert(branch["root"]["type"].is_string());
2✔
855
        std::string type = branch["root"]["type"];
2✔
856
        if (type == "sequence") {
2✔
857
            json_to_sequence(branch["root"], builder, branch_node);
2✔
858
        } else {
2✔
859
            throw std::runtime_error("Unknown child type");
×
860
        }
×
861
    }
2✔
862
}
1✔
863

864
void JSONSerializer::json_to_while_node(
865
    const nlohmann::json& j,
866
    builder::StructuredSDFGBuilder& builder,
867
    structured_control_flow::Sequence& parent,
868
    control_flow::Assignments& assignments
869
) {
4✔
870
    assert(j.contains("type"));
4✔
871
    assert(j["type"].is_string());
4✔
872
    assert(j["type"] == "while");
4✔
873
    assert(j.contains("root"));
4✔
874
    assert(j["root"].is_object());
4✔
875

876
    auto& while_node = builder.add_while(parent, assignments, json_to_debug_info(j["debug_info"]));
4✔
877
    while_node.element_id_ = j["element_id"];
4✔
878

879
    assert(j["root"]["type"] == "sequence");
4✔
880
    json_to_sequence(j["root"], builder, while_node.root());
4✔
881
}
4✔
882

883
void JSONSerializer::json_to_break_node(
884
    const nlohmann::json& j,
885
    builder::StructuredSDFGBuilder& builder,
886
    structured_control_flow::Sequence& parent,
887
    control_flow::Assignments& assignments
888
) {
1✔
889
    assert(j.contains("type"));
1✔
890
    assert(j["type"].is_string());
1✔
891
    assert(j["type"] == "break");
1✔
892
    auto& node = builder.add_break(parent, assignments, json_to_debug_info(j["debug_info"]));
1✔
893
    node.element_id_ = j["element_id"];
1✔
894
}
1✔
895

896
void JSONSerializer::json_to_continue_node(
897
    const nlohmann::json& j,
898
    builder::StructuredSDFGBuilder& builder,
899
    structured_control_flow::Sequence& parent,
900
    control_flow::Assignments& assignments
901
) {
1✔
902
    assert(j.contains("type"));
1✔
903
    assert(j["type"].is_string());
1✔
904
    assert(j["type"] == "continue");
1✔
905
    auto& node = builder.add_continue(parent, assignments, json_to_debug_info(j["debug_info"]));
1✔
906
    node.element_id_ = j["element_id"];
1✔
907
}
1✔
908

909
void JSONSerializer::json_to_map_node(
910
    const nlohmann::json& j,
911
    builder::StructuredSDFGBuilder& builder,
912
    structured_control_flow::Sequence& parent,
913
    control_flow::Assignments& assignments
914
) {
9✔
915
    assert(j.contains("type"));
9✔
916
    assert(j["type"].is_string());
9✔
917
    assert(j["type"] == "map");
9✔
918
    assert(j.contains("indvar"));
9✔
919
    assert(j["indvar"].is_string());
9✔
920
    assert(j.contains("init"));
9✔
921
    assert(j["init"].is_string());
9✔
922
    assert(j.contains("condition"));
9✔
923
    assert(j["condition"].is_string());
9✔
924
    assert(j.contains("update"));
9✔
925
    assert(j["update"].is_string());
9✔
926
    assert(j.contains("root"));
9✔
927
    assert(j["root"].is_object());
9✔
928
    assert(j.contains("schedule_type"));
9✔
929
    assert(j["schedule_type"].is_object());
9✔
930

931
    structured_control_flow::ScheduleType schedule_type = json_to_schedule_type(j["schedule_type"]);
9✔
932

933
    symbolic::Symbol indvar = symbolic::symbol(j["indvar"]);
9✔
934
    auto init = symbolic::parse(j["init"].get<std::string>());
9✔
935
    auto update = symbolic::parse(j["update"].get<std::string>());
9✔
936
    auto condition_expr = symbolic::parse(j["condition"].get<std::string>());
9✔
937
    symbolic::Condition condition = SymEngine::rcp_dynamic_cast<const SymEngine::Boolean>(condition_expr);
9✔
938
    if (condition.is_null()) {
9✔
939
        throw InvalidSDFGException("Map condition is not a boolean expression");
×
940
    }
×
941

942
    auto& map_node = builder.add_map(
9✔
943
        parent, indvar, condition, init, update, schedule_type, assignments, json_to_debug_info(j["debug_info"])
9✔
944
    );
9✔
945
    map_node.element_id_ = j["element_id"];
9✔
946

947
    assert(j["root"].contains("type"));
9✔
948
    assert(j["root"]["type"].is_string());
9✔
949
    assert(j["root"]["type"] == "sequence");
9✔
950
    json_to_sequence(j["root"], builder, map_node.root());
9✔
951
}
9✔
952

953
void JSONSerializer::json_to_return_node(
954
    const nlohmann::json& j,
955
    builder::StructuredSDFGBuilder& builder,
956
    structured_control_flow::Sequence& parent,
957
    control_flow::Assignments& assignments
958
) {
1✔
959
    assert(j.contains("type"));
1✔
960
    assert(j["type"].is_string());
1✔
961
    assert(j["type"] == "return");
1✔
962

963
    std::string data = j["data"];
1✔
964
    std::unique_ptr<types::IType> data_type = nullptr;
1✔
965
    if (j.contains("data_type")) {
1✔
966
        data_type = json_to_type(j["data_type"]);
×
967
    }
×
968

969
    if (data_type == nullptr) {
1✔
970
        auto& node = builder.add_return(parent, data, assignments, json_to_debug_info(j["debug_info"]));
1✔
971
        node.element_id_ = j["element_id"];
1✔
972
    } else {
1✔
973
        auto& node =
×
974
            builder.add_constant_return(parent, data, *data_type, assignments, json_to_debug_info(j["debug_info"]));
×
975
        node.element_id_ = j["element_id"];
×
976
    }
×
977
}
1✔
978

979
std::unique_ptr<types::IType> JSONSerializer::json_to_type(const nlohmann::json& j) {
520✔
980
    if (j.contains("type")) {
520✔
981
        if (j["type"] == "scalar") {
520✔
982
            // Deserialize scalar type
983
            assert(j.contains("primitive_type"));
374✔
984
            types::PrimitiveType primitive_type = j["primitive_type"];
374✔
985
            assert(j.contains("storage_type"));
374✔
986
            types::StorageType storage_type = json_to_storage_type(j["storage_type"]);
374✔
987
            assert(j.contains("initializer"));
374✔
988
            std::string initializer = j["initializer"];
374✔
989
            assert(j.contains("alignment"));
374✔
990
            size_t alignment = j["alignment"];
374✔
991
            return std::make_unique<types::Scalar>(storage_type, alignment, initializer, primitive_type);
374✔
992
        } else if (j["type"] == "array") {
374✔
993
            // Deserialize array type
994
            assert(j.contains("element_type"));
73✔
995
            std::unique_ptr<types::IType> member_type = json_to_type(j["element_type"]);
73✔
996
            assert(j.contains("num_elements"));
73✔
997
            std::string num_elements_str = j["num_elements"];
73✔
998
            // Convert num_elements_str to symbolic::Expression
999
            auto num_elements = symbolic::parse(num_elements_str);
73✔
1000
            assert(j.contains("storage_type"));
73✔
1001
            types::StorageType storage_type = json_to_storage_type(j["storage_type"]);
73✔
1002
            assert(j.contains("initializer"));
73✔
1003
            std::string initializer = j["initializer"];
73✔
1004
            assert(j.contains("alignment"));
73✔
1005
            size_t alignment = j["alignment"];
73✔
1006
            return std::make_unique<types::Array>(storage_type, alignment, initializer, *member_type, num_elements);
73✔
1007
        } else if (j["type"] == "pointer") {
73✔
1008
            // Deserialize pointer type
1009
            std::optional<std::unique_ptr<types::IType>> pointee_type;
62✔
1010
            if (j.contains("pointee_type")) {
62✔
1011
                assert(j.contains("pointee_type"));
49✔
1012
                pointee_type = json_to_type(j["pointee_type"]);
49✔
1013
            } else {
49✔
1014
                pointee_type = std::nullopt;
13✔
1015
            }
13✔
1016
            assert(j.contains("storage_type"));
62✔
1017
            types::StorageType storage_type = json_to_storage_type(j["storage_type"]);
62✔
1018
            assert(j.contains("initializer"));
62✔
1019
            std::string initializer = j["initializer"];
62✔
1020
            assert(j.contains("alignment"));
62✔
1021
            size_t alignment = j["alignment"];
62✔
1022
            if (pointee_type.has_value()) {
62✔
1023
                return std::make_unique<types::Pointer>(storage_type, alignment, initializer, *pointee_type.value());
49✔
1024
            } else {
49✔
1025
                return std::make_unique<types::Pointer>(storage_type, alignment, initializer);
13✔
1026
            }
13✔
1027
        } else if (j["type"] == "structure") {
62✔
1028
            // Deserialize structure type
1029
            assert(j.contains("name"));
3✔
1030
            std::string name = j["name"];
3✔
1031
            assert(j.contains("storage_type"));
3✔
1032
            types::StorageType storage_type = json_to_storage_type(j["storage_type"]);
3✔
1033
            assert(j.contains("initializer"));
3✔
1034
            std::string initializer = j["initializer"];
3✔
1035
            assert(j.contains("alignment"));
3✔
1036
            size_t alignment = j["alignment"];
3✔
1037
            return std::make_unique<types::Structure>(storage_type, alignment, initializer, name);
3✔
1038
        } else if (j["type"] == "function") {
8✔
1039
            // Deserialize function type
1040
            assert(j.contains("return_type"));
4✔
1041
            std::unique_ptr<types::IType> return_type = json_to_type(j["return_type"]);
4✔
1042
            assert(j.contains("params"));
4✔
1043
            std::vector<std::unique_ptr<types::IType>> params;
4✔
1044
            for (const auto& param : j["params"]) {
10✔
1045
                params.push_back(json_to_type(param));
10✔
1046
            }
10✔
1047
            assert(j.contains("is_var_arg"));
4✔
1048
            bool is_var_arg = j["is_var_arg"];
4✔
1049
            assert(j.contains("storage_type"));
4✔
1050
            types::StorageType storage_type = json_to_storage_type(j["storage_type"]);
4✔
1051
            assert(j.contains("initializer"));
4✔
1052
            std::string initializer = j["initializer"];
4✔
1053
            assert(j.contains("alignment"));
4✔
1054
            size_t alignment = j["alignment"];
4✔
1055
            auto function =
4✔
1056
                std::make_unique<types::Function>(storage_type, alignment, initializer, *return_type, is_var_arg);
4✔
1057
            for (const auto& param : params) {
10✔
1058
                function->add_param(*param);
10✔
1059
            }
10✔
1060
            return function->clone();
4✔
1061
        } else if (j["type"] == "reference") {
4✔
1062
            // Deserialize reference type
1063
            assert(j.contains("reference_type"));
4✔
1064
            std::unique_ptr<types::IType> reference_type = json_to_type(j["reference_type"]);
4✔
1065
            return std::make_unique<sdfg::codegen::Reference>(*reference_type);
4✔
1066
        } else if (j["type"] == "tensor") {
4✔
1067
            // Deserialize tensor type
1068
            assert(j.contains("element_type"));
×
1069
            std::unique_ptr<types::IType> element_type = json_to_type(j["element_type"]);
×
1070
            assert(j.contains("shape"));
×
1071
            std::vector<symbolic::Expression> shape;
×
1072
            for (const auto& dim : j["shape"]) {
×
1073
                assert(dim.is_string());
×
1074
                std::string dim_str = dim;
×
1075
                auto expr = symbolic::parse(dim_str);
×
1076
                shape.push_back(expr);
×
1077
            }
×
1078
            assert(j.contains("strides"));
×
1079
            std::vector<symbolic::Expression> strides;
×
1080
            for (const auto& stride : j["strides"]) {
×
1081
                assert(stride.is_string());
×
1082
                std::string stride_str = stride;
×
1083
                auto expr = symbolic::parse(stride_str);
×
1084
                strides.push_back(expr);
×
1085
            }
×
1086
            assert(j.contains("offset"));
×
1087
            symbolic::Expression offset = symbolic::parse(j["offset"].get<std::string>());
×
1088
            assert(j.contains("storage_type"));
×
1089
            types::StorageType storage_type = json_to_storage_type(j["storage_type"]);
×
1090
            assert(j.contains("initializer"));
×
1091
            std::string initializer = j["initializer"];
×
1092
            assert(j.contains("alignment"));
×
1093
            size_t alignment = j["alignment"];
×
1094
            return std::make_unique<types::Tensor>(
×
1095
                storage_type, alignment, initializer, dynamic_cast<types::Scalar&>(*element_type), shape, strides, offset
×
1096
            );
×
1097
        } else {
×
1098
            throw std::runtime_error("Unknown type");
×
1099
        }
×
1100
    } else {
520✔
1101
        throw std::runtime_error("Type not found");
×
1102
    }
×
1103
}
520✔
1104

1105
DebugInfo JSONSerializer::json_to_debug_info(const nlohmann::json& j) {
616✔
1106
    assert(j.contains("has"));
616✔
1107
    assert(j["has"].is_boolean());
616✔
1108
    if (!j["has"]) {
616✔
1109
        return DebugInfo();
612✔
1110
    }
612✔
1111
    assert(j.contains("filename"));
616✔
1112
    assert(j["filename"].is_string());
4✔
1113
    std::string filename = j["filename"];
4✔
1114
    assert(j.contains("function"));
4✔
1115
    assert(j["function"].is_string());
4✔
1116
    std::string function = j["function"];
4✔
1117
    assert(j.contains("start_line"));
4✔
1118
    assert(j["start_line"].is_number_integer());
4✔
1119
    size_t start_line = j["start_line"];
4✔
1120
    assert(j.contains("start_column"));
4✔
1121
    assert(j["start_column"].is_number_integer());
4✔
1122
    size_t start_column = j["start_column"];
4✔
1123
    assert(j.contains("end_line"));
4✔
1124
    assert(j["end_line"].is_number_integer());
4✔
1125
    size_t end_line = j["end_line"];
4✔
1126
    assert(j.contains("end_column"));
4✔
1127
    assert(j["end_column"].is_number_integer());
4✔
1128
    size_t end_column = j["end_column"];
4✔
1129
    return DebugInfo(filename, function, start_line, start_column, end_line, end_column);
4✔
1130
}
4✔
1131

1132
ScheduleType JSONSerializer::json_to_schedule_type(const nlohmann::json& j) {
10✔
1133
    assert(j.contains("value"));
10✔
1134
    assert(j["value"].is_string());
10✔
1135
    // assert(j.contains("category"));
1136
    // assert(j["category"].is_number_integer());
1137
    assert(j.contains("properties"));
10✔
1138
    assert(j["properties"].is_object());
10✔
1139
    ScheduleTypeCategory category = ScheduleTypeCategory::None;
10✔
1140
    if (j.contains("category")) {
10✔
1141
        category = static_cast<ScheduleTypeCategory>(j["category"].get<int>());
10✔
1142
    }
10✔
1143
    ScheduleType schedule_type(j["value"].get<std::string>(), category);
10✔
1144
    for (const auto& [key, value] : j["properties"].items()) {
11✔
1145
        assert(value.is_string());
11✔
1146
        schedule_type.set_property(key, value.get<std::string>());
11✔
1147
    }
11✔
1148
    return schedule_type;
10✔
1149
}
10✔
1150

1151
types::StorageType JSONSerializer::json_to_storage_type(const nlohmann::json& j) {
516✔
1152
    if (!j.contains("value")) {
516✔
1153
        return types::StorageType::CPU_Stack();
×
1154
    }
×
1155
    std::string value = j["value"].get<std::string>();
516✔
1156

1157
    symbolic::Expression allocation_size = SymEngine::null;
516✔
1158
    if (j.contains("allocation_size")) {
516✔
1159
        allocation_size = symbolic::parse(j["allocation_size"].get<std::string>());
6✔
1160
    }
6✔
1161

1162
    types::StorageType::AllocationType allocation = j["allocation"];
516✔
1163
    types::StorageType::AllocationType deallocation = j["deallocation"];
516✔
1164

1165
    auto storageType = types::StorageType(j["value"].get<std::string>(), allocation_size, allocation, deallocation);
516✔
1166

1167
    if (j.contains("args")) {
516✔
1168
        nlohmann::json::array_t args = j["args"];
×
1169
        if (args.size() > 0) {
×
1170
            storageType.arg1(symbolic::parse(args[0].get<std::string>()));
×
1171
        }
×
1172
    }
×
1173
    return storageType;
516✔
1174
}
516✔
1175

1176
std::string JSONSerializer::expression(const symbolic::Expression expr) {
321✔
1177
    JSONSymbolicPrinter printer;
321✔
1178
    return printer.apply(expr);
321✔
1179
};
321✔
1180

1181
void JSONSerializer::writeToFile(const StructuredSDFG& sdfg, const std::filesystem::path& file) {
×
1182
    JSONSerializer ser;
×
1183
    auto json = ser.serialize(sdfg);
×
1184

UNCOV
1185
    auto parent_path = file.parent_path();
×
1186
    if (!parent_path.empty()) {
×
1187
        std::filesystem::create_directories(file.parent_path());
×
1188
    }
×
1189

UNCOV
1190
    std::ofstream out(file, std::ofstream::out);
×
UNCOV
1191
    if (!out.is_open()) {
×
UNCOV
1192
        std::cerr << "Could not open file " << file << " for writing JSON output." << std::endl;
×
UNCOV
1193
    }
×
UNCOV
1194
    out << json << std::endl;
×
UNCOV
1195
    out.close();
×
UNCOV
1196
}
×
1197

UNCOV
1198
void JSONSymbolicPrinter::bvisit(const SymEngine::Equality& x) {
×
UNCOV
1199
    str_ = apply(x.get_args()[0]) + " == " + apply(x.get_args()[1]);
×
UNCOV
1200
    str_ = parenthesize(str_);
×
1201
};
×
1202

1203
void JSONSymbolicPrinter::bvisit(const SymEngine::Unequality& x) {
×
1204
    str_ = apply(x.get_args()[0]) + " != " + apply(x.get_args()[1]);
×
1205
    str_ = parenthesize(str_);
×
1206
};
×
1207

1208
void JSONSymbolicPrinter::bvisit(const SymEngine::LessThan& x) {
2✔
1209
    str_ = apply(x.get_args()[0]) + " <= " + apply(x.get_args()[1]);
2✔
1210
    str_ = parenthesize(str_);
2✔
1211
};
2✔
1212

1213
void JSONSymbolicPrinter::bvisit(const SymEngine::StrictLessThan& x) {
26✔
1214
    str_ = apply(x.get_args()[0]) + " < " + apply(x.get_args()[1]);
26✔
1215
    str_ = parenthesize(str_);
26✔
1216
};
26✔
1217

1218
void JSONSymbolicPrinter::bvisit(const SymEngine::Min& x) {
×
1219
    std::ostringstream s;
×
UNCOV
1220
    auto container = x.get_args();
×
1221
    if (container.size() == 1) {
×
1222
        s << apply(*container.begin());
×
UNCOV
1223
    } else {
×
1224
        s << "min(";
×
1225
        s << apply(*container.begin());
×
1226

1227
        // Recursively apply __daisy_min to the arguments
1228
        SymEngine::vec_basic subargs;
×
1229
        for (auto it = ++(container.begin()); it != container.end(); ++it) {
×
1230
            subargs.push_back(*it);
×
1231
        }
×
UNCOV
1232
        auto submin = SymEngine::min(subargs);
×
UNCOV
1233
        s << ", " << apply(submin);
×
1234

1235
        s << ")";
×
1236
    }
×
1237

1238
    str_ = s.str();
×
1239
};
×
1240

1241
void JSONSymbolicPrinter::bvisit(const SymEngine::Max& x) {
×
1242
    std::ostringstream s;
×
UNCOV
1243
    auto container = x.get_args();
×
1244
    if (container.size() == 1) {
×
1245
        s << apply(*container.begin());
×
UNCOV
1246
    } else {
×
UNCOV
1247
        s << "max(";
×
UNCOV
1248
        s << apply(*container.begin());
×
1249

1250
        // Recursively apply __daisy_max to the arguments
1251
        SymEngine::vec_basic subargs;
×
1252
        for (auto it = ++(container.begin()); it != container.end(); ++it) {
×
UNCOV
1253
            subargs.push_back(*it);
×
UNCOV
1254
        }
×
UNCOV
1255
        auto submax = SymEngine::max(subargs);
×
UNCOV
1256
        s << ", " << apply(submax);
×
1257

UNCOV
1258
        s << ")";
×
UNCOV
1259
    }
×
1260

1261
    str_ = s.str();
×
UNCOV
1262
};
×
1263

1264
void LibraryNodeSerializerRegistry::
1265
    register_library_node_serializer(std::string library_node_code, LibraryNodeSerializerFn fn) {
154✔
1266
    std::lock_guard<std::mutex> lock(mutex_);
154✔
1267
    if (factory_map_.find(library_node_code) != factory_map_.end()) {
154✔
UNCOV
1268
        return;
×
UNCOV
1269
    }
×
1270
    factory_map_[library_node_code] = std::move(fn);
154✔
1271
}
154✔
1272

1273
LibraryNodeSerializerFn LibraryNodeSerializerRegistry::get_library_node_serializer(std::string library_node_code) {
25✔
1274
    auto it = factory_map_.find(library_node_code);
25✔
1275
    if (it != factory_map_.end()) {
25✔
1276
        return it->second;
25✔
1277
    }
25✔
1278
    return nullptr;
×
1279
}
25✔
1280

UNCOV
1281
size_t LibraryNodeSerializerRegistry::size() const { return factory_map_.size(); }
×
1282

1283
void register_default_serializers() {
3✔
1284
    // stdlib
1285
    LibraryNodeSerializerRegistry::instance()
3✔
1286
        .register_library_node_serializer(stdlib::LibraryNodeType_Alloca.value(), []() {
3✔
1287
            return std::make_unique<stdlib::AllocaNodeSerializer>();
×
UNCOV
1288
        });
×
1289
    LibraryNodeSerializerRegistry::instance()
3✔
1290
        .register_library_node_serializer(stdlib::LibraryNodeType_Assert.value(), []() {
3✔
1291
            return std::make_unique<stdlib::AssertNodeSerializer>();
×
UNCOV
1292
        });
×
1293
    LibraryNodeSerializerRegistry::instance()
3✔
1294
        .register_library_node_serializer(stdlib::LibraryNodeType_Calloc.value(), []() {
3✔
1295
            return std::make_unique<stdlib::CallocNodeSerializer>();
×
UNCOV
1296
        });
×
1297
    LibraryNodeSerializerRegistry::instance()
3✔
1298
        .register_library_node_serializer(stdlib::LibraryNodeType_Free.value(), []() {
3✔
1299
            return std::make_unique<stdlib::FreeNodeSerializer>();
×
UNCOV
1300
        });
×
1301
    LibraryNodeSerializerRegistry::instance()
3✔
1302
        .register_library_node_serializer(stdlib::LibraryNodeType_Malloc.value(), []() {
3✔
1303
            return std::make_unique<stdlib::MallocNodeSerializer>();
×
UNCOV
1304
        });
×
1305
    LibraryNodeSerializerRegistry::instance()
3✔
1306
        .register_library_node_serializer(stdlib::LibraryNodeType_Memcpy.value(), []() {
3✔
1307
            return std::make_unique<stdlib::MemcpyNodeSerializer>();
×
UNCOV
1308
        });
×
1309
    LibraryNodeSerializerRegistry::instance()
3✔
1310
        .register_library_node_serializer(stdlib::LibraryNodeType_Memmove.value(), []() {
3✔
UNCOV
1311
            return std::make_unique<stdlib::MemmoveNodeSerializer>();
×
1312
        });
×
1313
    LibraryNodeSerializerRegistry::instance()
3✔
1314
        .register_library_node_serializer(stdlib::LibraryNodeType_Memset.value(), []() {
3✔
UNCOV
1315
            return std::make_unique<stdlib::MemsetNodeSerializer>();
×
UNCOV
1316
        });
×
1317
    LibraryNodeSerializerRegistry::instance()
3✔
1318
        .register_library_node_serializer(stdlib::LibraryNodeType_Trap.value(), []() {
3✔
UNCOV
1319
            return std::make_unique<stdlib::TrapNodeSerializer>();
×
UNCOV
1320
        });
×
1321
    LibraryNodeSerializerRegistry::instance()
3✔
1322
        .register_library_node_serializer(stdlib::LibraryNodeType_Unreachable.value(), []() {
3✔
UNCOV
1323
            return std::make_unique<stdlib::UnreachableNodeSerializer>();
×
UNCOV
1324
        });
×
1325

1326
    // Metadata
1327
    LibraryNodeSerializerRegistry::instance()
3✔
1328
        .register_library_node_serializer(data_flow::LibraryNodeType_Metadata.value(), []() {
3✔
1329
            return std::make_unique<data_flow::MetadataNodeSerializer>();
×
UNCOV
1330
        });
×
1331

1332
    // Barrier
1333
    LibraryNodeSerializerRegistry::instance()
3✔
1334
        .register_library_node_serializer(data_flow::LibraryNodeType_BarrierLocal.value(), []() {
3✔
1335
            return std::make_unique<data_flow::BarrierLocalNodeSerializer>();
1✔
1336
        });
1✔
1337

1338
    // Call Node
1339
    LibraryNodeSerializerRegistry::instance()
3✔
1340
        .register_library_node_serializer(data_flow::LibraryNodeType_Call.value(), []() {
6✔
1341
            return std::make_unique<data_flow::CallNodeSerializer>();
6✔
1342
        });
6✔
1343
    LibraryNodeSerializerRegistry::instance()
3✔
1344
        .register_library_node_serializer(data_flow::LibraryNodeType_Invoke.value(), []() {
3✔
1345
            return std::make_unique<data_flow::InvokeNodeSerializer>();
×
1346
        });
×
1347

1348
    // LoadConst
1349
    LibraryNodeSerializerRegistry::instance()
3✔
1350
        .register_library_node_serializer(data_flow::LibraryNodeType_LoadConst.value(), []() {
3✔
1351
            return std::make_unique<data_flow::LoadConstNodeSerializer>();
2✔
1352
        });
2✔
1353

1354
    // CMath
1355
    LibraryNodeSerializerRegistry::instance()
3✔
1356
        .register_library_node_serializer(math::cmath::LibraryNodeType_CMath.value(), []() {
3✔
UNCOV
1357
            return std::make_unique<math::cmath::CMathNodeSerializer>();
×
UNCOV
1358
        });
×
1359
    // Backward compatibility
1360
    LibraryNodeSerializerRegistry::instance()
3✔
1361
        .register_library_node_serializer(math::cmath::LibraryNodeType_CMath_Deprecated.value(), []() {
3✔
1362
            return std::make_unique<math::cmath::CMathNodeSerializer>();
×
1363
        });
×
1364

1365
    // BLAS
1366
    LibraryNodeSerializerRegistry::instance()
3✔
1367
        .register_library_node_serializer(math::blas::LibraryNodeType_DOT.value(), []() {
3✔
UNCOV
1368
            return std::make_unique<math::blas::DotNodeSerializer>();
×
UNCOV
1369
        });
×
1370
    LibraryNodeSerializerRegistry::instance()
3✔
1371
        .register_library_node_serializer(math::blas::LibraryNodeType_GEMM.value(), []() {
3✔
UNCOV
1372
            return std::make_unique<math::blas::GEMMNodeSerializer>();
×
UNCOV
1373
        });
×
1374

1375
    // Tensor
1376

1377
    LibraryNodeSerializerRegistry::instance()
3✔
1378
        .register_library_node_serializer(math::tensor::LibraryNodeType_Broadcast.value(), []() {
3✔
UNCOV
1379
            return std::make_unique<math::tensor::BroadcastNodeSerializer>();
×
1380
        });
×
1381
    LibraryNodeSerializerRegistry::instance()
3✔
1382
        .register_library_node_serializer(math::tensor::LibraryNodeType_Conv.value(), []() {
3✔
UNCOV
1383
            return std::make_unique<math::tensor::ConvNodeSerializer>();
×
1384
        });
×
1385
    LibraryNodeSerializerRegistry::instance()
3✔
1386
        .register_library_node_serializer(math::tensor::LibraryNodeType_Pooling.value(), []() {
3✔
UNCOV
1387
            return std::make_unique<math::tensor::PoolingNodeSerializer>();
×
1388
        });
×
1389
    LibraryNodeSerializerRegistry::instance()
3✔
1390
        .register_library_node_serializer(math::tensor::LibraryNodeType_Transpose.value(), []() {
3✔
UNCOV
1391
            return std::make_unique<math::tensor::TransposeNodeSerializer>();
×
1392
        });
×
1393
    LibraryNodeSerializerRegistry::instance()
3✔
1394
        .register_library_node_serializer(math::tensor::LibraryNodeType_MatMul.value(), []() {
3✔
UNCOV
1395
            return std::make_unique<math::tensor::MatMulNodeSerializer>();
×
1396
        });
×
1397

1398
    // Elementwise
1399
    LibraryNodeSerializerRegistry::instance()
3✔
1400
        .register_library_node_serializer(math::tensor::LibraryNodeType_Abs.value(), []() {
3✔
1401
            return std::make_unique<math::tensor::AbsNodeSerializer>();
×
UNCOV
1402
        });
×
1403
    LibraryNodeSerializerRegistry::instance()
3✔
1404
        .register_library_node_serializer(math::tensor::LibraryNodeType_Add.value(), []() {
3✔
1405
            return std::make_unique<math::tensor::AddNodeSerializer>();
×
UNCOV
1406
        });
×
1407
    LibraryNodeSerializerRegistry::instance()
3✔
1408
        .register_library_node_serializer(math::tensor::LibraryNodeType_Div.value(), []() {
3✔
1409
            return std::make_unique<math::tensor::DivNodeSerializer>();
×
UNCOV
1410
        });
×
1411
    LibraryNodeSerializerRegistry::instance()
3✔
1412
        .register_library_node_serializer(math::tensor::LibraryNodeType_Elu.value(), []() {
3✔
1413
            return std::make_unique<math::tensor::EluNodeSerializer>();
×
UNCOV
1414
        });
×
1415
    LibraryNodeSerializerRegistry::instance()
3✔
1416
        .register_library_node_serializer(math::tensor::LibraryNodeType_Exp.value(), []() {
3✔
1417
            return std::make_unique<math::tensor::ExpNodeSerializer>();
×
UNCOV
1418
        });
×
1419
    LibraryNodeSerializerRegistry::instance()
3✔
1420
        .register_library_node_serializer(math::tensor::LibraryNodeType_Erf.value(), []() {
3✔
1421
            return std::make_unique<math::tensor::ErfNodeSerializer>();
×
UNCOV
1422
        });
×
1423
    LibraryNodeSerializerRegistry::instance()
3✔
1424
        .register_library_node_serializer(math::tensor::LibraryNodeType_HardSigmoid.value(), []() {
3✔
1425
            return std::make_unique<math::tensor::HardSigmoidNodeSerializer>();
×
UNCOV
1426
        });
×
1427
    LibraryNodeSerializerRegistry::instance()
3✔
1428
        .register_library_node_serializer(math::tensor::LibraryNodeType_LeakyReLU.value(), []() {
3✔
1429
            return std::make_unique<math::tensor::LeakyReLUNodeSerializer>();
×
UNCOV
1430
        });
×
1431
    LibraryNodeSerializerRegistry::instance()
3✔
1432
        .register_library_node_serializer(math::tensor::LibraryNodeType_Mul.value(), []() {
3✔
1433
            return std::make_unique<math::tensor::MulNodeSerializer>();
×
UNCOV
1434
        });
×
1435
    LibraryNodeSerializerRegistry::instance()
3✔
1436
        .register_library_node_serializer(math::tensor::LibraryNodeType_Pow.value(), []() {
3✔
1437
            return std::make_unique<math::tensor::PowNodeSerializer>();
×
UNCOV
1438
        });
×
1439
    LibraryNodeSerializerRegistry::instance()
3✔
1440
        .register_library_node_serializer(math::tensor::LibraryNodeType_ReLU.value(), []() {
3✔
1441
            return std::make_unique<math::tensor::ReLUNodeSerializer>();
×
UNCOV
1442
        });
×
1443
    LibraryNodeSerializerRegistry::instance()
3✔
1444
        .register_library_node_serializer(math::tensor::LibraryNodeType_Sigmoid.value(), []() {
3✔
1445
            return std::make_unique<math::tensor::SigmoidNodeSerializer>();
×
UNCOV
1446
        });
×
1447
    LibraryNodeSerializerRegistry::instance()
3✔
1448
        .register_library_node_serializer(math::tensor::LibraryNodeType_Sqrt.value(), []() {
3✔
1449
            return std::make_unique<math::tensor::SqrtNodeSerializer>();
×
UNCOV
1450
        });
×
1451
    LibraryNodeSerializerRegistry::instance()
3✔
1452
        .register_library_node_serializer(math::tensor::LibraryNodeType_Sub.value(), []() {
3✔
1453
            return std::make_unique<math::tensor::SubNodeSerializer>();
×
UNCOV
1454
        });
×
1455
    LibraryNodeSerializerRegistry::instance()
3✔
1456
        .register_library_node_serializer(math::tensor::LibraryNodeType_Tanh.value(), []() {
3✔
1457
            return std::make_unique<math::tensor::TanhNodeSerializer>();
×
UNCOV
1458
        });
×
1459
    LibraryNodeSerializerRegistry::instance()
3✔
1460
        .register_library_node_serializer(math::tensor::LibraryNodeType_Minimum.value(), []() {
3✔
UNCOV
1461
            return std::make_unique<math::tensor::MinimumNodeSerializer>();
×
1462
        });
×
1463
    LibraryNodeSerializerRegistry::instance()
3✔
1464
        .register_library_node_serializer(math::tensor::LibraryNodeType_Maximum.value(), []() {
3✔
UNCOV
1465
            return std::make_unique<math::tensor::MaximumNodeSerializer>();
×
1466
        });
×
1467
    LibraryNodeSerializerRegistry::instance()
3✔
1468
        .register_library_node_serializer(math::tensor::LibraryNodeType_Fill.value(), []() {
3✔
UNCOV
1469
            return std::make_unique<math::tensor::FillNodeSerializer>();
×
1470
        });
×
1471
    LibraryNodeSerializerRegistry::instance()
3✔
1472
        .register_library_node_serializer(math::tensor::LibraryNodeType_TensorTasklet.value(), []() {
3✔
UNCOV
1473
            return std::make_unique<math::tensor::TaskletTensorNodeSerializer>();
×
1474
        });
×
1475
    LibraryNodeSerializerRegistry::instance()
3✔
1476
        .register_library_node_serializer(math::tensor::LibraryNodeType_TensorCMath.value(), []() {
3✔
UNCOV
1477
            return std::make_unique<math::tensor::CMathTensorNodeSerializer>();
×
1478
        });
×
1479
    LibraryNodeSerializerRegistry::instance()
3✔
1480
        .register_library_node_serializer(math::tensor::LibraryNodeType_Cast.value(), []() {
3✔
UNCOV
1481
            return std::make_unique<math::tensor::CastNodeSerializer>();
×
1482
        });
×
1483

1484
    // Reduce
1485
    LibraryNodeSerializerRegistry::instance()
3✔
1486
        .register_library_node_serializer(math::tensor::LibraryNodeType_Sum.value(), []() {
3✔
UNCOV
1487
            return std::make_unique<math::tensor::SumNodeSerializer>();
×
UNCOV
1488
        });
×
1489
    LibraryNodeSerializerRegistry::instance()
3✔
1490
        .register_library_node_serializer(math::tensor::LibraryNodeType_Max.value(), []() {
3✔
UNCOV
1491
            return std::make_unique<math::tensor::MaxNodeSerializer>();
×
UNCOV
1492
        });
×
1493
    LibraryNodeSerializerRegistry::instance()
3✔
1494
        .register_library_node_serializer(math::tensor::LibraryNodeType_Min.value(), []() {
3✔
UNCOV
1495
            return std::make_unique<math::tensor::MinNodeSerializer>();
×
UNCOV
1496
        });
×
1497
    LibraryNodeSerializerRegistry::instance()
3✔
1498
        .register_library_node_serializer(math::tensor::LibraryNodeType_Softmax.value(), []() {
3✔
UNCOV
1499
            return std::make_unique<math::tensor::SoftmaxNodeSerializer>();
×
UNCOV
1500
        });
×
1501
    LibraryNodeSerializerRegistry::instance()
3✔
1502
        .register_library_node_serializer(math::tensor::LibraryNodeType_Mean.value(), []() {
3✔
UNCOV
1503
            return std::make_unique<math::tensor::MeanNodeSerializer>();
×
UNCOV
1504
        });
×
1505
    LibraryNodeSerializerRegistry::instance()
3✔
1506
        .register_library_node_serializer(math::tensor::LibraryNodeType_Std.value(), []() {
3✔
UNCOV
1507
            return std::make_unique<math::tensor::StdNodeSerializer>();
×
UNCOV
1508
        });
×
1509
}
3✔
1510

1511
} // namespace serializer
1512
} // 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