GCC Code Coverage Report


Directory: ../
File: src/irgenerator/GenExpressions.cpp
Date: 2025-10-09 06:28:01
Coverage Exec Excl Total
Lines: 98.8% 501 17 524
Functions: 95.0% 19 0 20
Branches: 59.4% 550 36 962

Line Branch Exec Source
1 // Copyright (c) 2021-2025 ChilliBits. All rights reserved.
2
3 #include "IRGenerator.h"
4
5 #include <ast/ASTNodes.h>
6
7 #include <llvm/IR/Module.h>
8
9 namespace spice::compiler {
10
11 65843 std::any IRGenerator::visitAssignExpr(const AssignExprNode *node) {
12 65843 diGenerator.setSourceLocation(node);
13
14 // Visit ternary expression
15
2/2
✓ Branch 3 → 4 taken 59893 times.
✓ Branch 3 → 5 taken 5950 times.
65843 if (node->ternaryExpr)
16 59893 return visit(node->ternaryExpr);
17
18 // Assign or compound assign operation
19
1/2
✓ Branch 5 → 6 taken 5950 times.
✗ Branch 5 → 60 not taken.
5950 if (node->op != AssignExprNode::AssignOp::OP_NONE) {
20 5950 const PrefixUnaryExprNode *lhsNode = node->lhs;
21 5950 const AssignExprNode *rhsNode = node->rhs;
22
23 // Normal assignment
24
2/2
✓ Branch 6 → 7 taken 5282 times.
✓ Branch 6 → 11 taken 668 times.
5950 if (node->op == AssignExprNode::AssignOp::OP_ASSIGN)
25
2/4
✓ Branch 7 → 8 taken 5282 times.
✗ Branch 7 → 69 not taken.
✓ Branch 8 → 9 taken 5282 times.
✗ Branch 8 → 69 not taken.
5282 return doAssignment(lhsNode, rhsNode, node);
26
27 // Compound assignment
28 // Get symbol types of left and right side
29
1/2
✓ Branch 11 → 12 taken 668 times.
✗ Branch 11 → 85 not taken.
668 const QualType lhsSTy = lhsNode->getEvaluatedSymbolType(manIdx);
30
1/2
✓ Branch 12 → 13 taken 668 times.
✗ Branch 12 → 85 not taken.
668 const QualType rhsSTy = rhsNode->getEvaluatedSymbolType(manIdx);
31
32 // Retrieve rhs
33
2/4
✓ Branch 13 → 14 taken 668 times.
✗ Branch 13 → 72 not taken.
✓ Branch 14 → 15 taken 668 times.
✗ Branch 14 → 70 not taken.
668 auto rhs = std::any_cast<LLVMExprResult>(visit(rhsNode));
34 // Retrieve lhs
35
2/4
✓ Branch 16 → 17 taken 668 times.
✗ Branch 16 → 75 not taken.
✓ Branch 17 → 18 taken 668 times.
✗ Branch 17 → 73 not taken.
668 auto lhs = std::any_cast<LLVMExprResult>(visit(lhsNode));
36
37 668 LLVMExprResult result;
38
10/11
✓ Branch 19 → 20 taken 243 times.
✓ Branch 19 → 22 taken 32 times.
✓ Branch 19 → 24 taken 39 times.
✓ Branch 19 → 26 taken 44 times.
✓ Branch 19 → 28 taken 5 times.
✓ Branch 19 → 30 taken 1 time.
✓ Branch 19 → 32 taken 2 times.
✓ Branch 19 → 34 taken 1 time.
✓ Branch 19 → 36 taken 1 time.
✓ Branch 19 → 38 taken 300 times.
✗ Branch 19 → 40 not taken.
668 switch (node->op) {
39 243 case AssignExprNode::AssignOp::OP_PLUS_EQUAL:
40
1/2
✓ Branch 20 → 21 taken 243 times.
✗ Branch 20 → 85 not taken.
243 result = conversionManager.getPlusEqualInst(node, lhs, lhsSTy, rhs, rhsSTy, 0);
41 243 break;
42 32 case AssignExprNode::AssignOp::OP_MINUS_EQUAL:
43
1/2
✓ Branch 22 → 23 taken 32 times.
✗ Branch 22 → 85 not taken.
32 result = conversionManager.getMinusEqualInst(node, lhs, lhsSTy, rhs, rhsSTy, 0);
44 32 break;
45 39 case AssignExprNode::AssignOp::OP_MUL_EQUAL:
46
1/2
✓ Branch 24 → 25 taken 39 times.
✗ Branch 24 → 85 not taken.
39 result = conversionManager.getMulEqualInst(node, lhs, lhsSTy, rhs, rhsSTy, 0);
47 39 break;
48 44 case AssignExprNode::AssignOp::OP_DIV_EQUAL:
49
1/2
✓ Branch 26 → 27 taken 44 times.
✗ Branch 26 → 85 not taken.
44 result = conversionManager.getDivEqualInst(node, lhs, lhsSTy, rhs, rhsSTy, 0);
50 44 break;
51 5 case AssignExprNode::AssignOp::OP_REM_EQUAL:
52
1/2
✓ Branch 28 → 29 taken 5 times.
✗ Branch 28 → 85 not taken.
5 result = conversionManager.getRemEqualInst(node, lhs, lhsSTy, rhs, rhsSTy);
53 5 break;
54 1 case AssignExprNode::AssignOp::OP_SHL_EQUAL:
55
1/2
✓ Branch 30 → 31 taken 1 time.
✗ Branch 30 → 85 not taken.
1 result = conversionManager.getSHLEqualInst(node, lhs, lhsSTy, rhs, rhsSTy);
56 1 break;
57 2 case AssignExprNode::AssignOp::OP_SHR_EQUAL:
58
1/2
✓ Branch 32 → 33 taken 2 times.
✗ Branch 32 → 85 not taken.
2 result = conversionManager.getSHREqualInst(node, lhs, lhsSTy, rhs, rhsSTy);
59 2 break;
60 1 case AssignExprNode::AssignOp::OP_AND_EQUAL:
61
1/2
✓ Branch 34 → 35 taken 1 time.
✗ Branch 34 → 85 not taken.
1 result = conversionManager.getAndEqualInst(node, lhs, lhsSTy, rhs, rhsSTy);
62 1 break;
63 1 case AssignExprNode::AssignOp::OP_OR_EQUAL:
64
1/2
✓ Branch 36 → 37 taken 1 time.
✗ Branch 36 → 85 not taken.
1 result = conversionManager.getOrEqualInst(node, lhs, lhsSTy, rhs, rhsSTy);
65 1 break;
66 300 case AssignExprNode::AssignOp::OP_XOR_EQUAL:
67
1/2
✓ Branch 38 → 39 taken 300 times.
✗ Branch 38 → 85 not taken.
300 result = conversionManager.getXorEqualInst(node, lhs, lhsSTy, rhs, rhsSTy);
68 300 break;
69 default: // GCOV_EXCL_LINE
70 throw CompilerError(UNHANDLED_BRANCH, "Assign op fall-through"); // GCOV_EXCL_LINE
71 }
72
73
1/2
✗ Branch 48 → 49 not taken.
✓ Branch 48 → 51 taken 668 times.
668 if (result.ptr) { // The operation allocated more memory
74 if (lhs.entry)
75 lhs.entry->updateAddress(result.ptr);
76
2/2
✓ Branch 51 → 52 taken 525 times.
✓ Branch 51 → 57 taken 143 times.
668 } else if (result.value) { // The operation only updated the value
77 // Store the result
78 525 lhs.value = result.value;
79
5/6
✓ Branch 52 → 53 taken 287 times.
✓ Branch 52 → 55 taken 238 times.
✓ Branch 53 → 54 taken 2 times.
✓ Branch 53 → 55 taken 285 times.
✓ Branch 56 → 57 taken 525 times.
✗ Branch 56 → 85 not taken.
525 insertStore(lhs.value, lhs.ptr, lhs.entry && lhs.entry->isVolatile);
80 }
81
1/2
✓ Branch 57 → 58 taken 668 times.
✗ Branch 57 → 85 not taken.
668 return lhs;
82 }
83
84 // This is a fallthrough case -> throw an error
85 throw CompilerError(UNHANDLED_BRANCH, "AssignStmt fall-through"); // GCOV_EXCL_LINE
86 }
87
88 60172 std::any IRGenerator::visitTernaryExpr(const TernaryExprNode *node) {
89 60172 diGenerator.setSourceLocation(node);
90
91 // Check if only one operand is present -> loop through
92
2/2
✓ Branch 3 → 4 taken 59715 times.
✓ Branch 3 → 5 taken 457 times.
60172 if (!node->falseExpr)
93 59715 return visit(node->condition);
94
95 // It is a ternary
96 // Retrieve the condition value
97 457 llvm::Value *condValue = resolveValue(node->condition);
98
2/2
✓ Branch 6 → 7 taken 1 time.
✓ Branch 6 → 8 taken 456 times.
457 const LogicalOrExprNode *trueNode = node->isShortened ? node->condition : node->trueExpr;
99 457 const LogicalOrExprNode *falseNode = node->falseExpr;
100
101 457 llvm::Value *resultValue = nullptr;
102 457 llvm::Value *resultPtr = nullptr;
103 457 SymbolTableEntry *anonymousSymbol = nullptr;
104
6/6
✓ Branch 10 → 11 taken 126 times.
✓ Branch 10 → 14 taken 331 times.
✓ Branch 12 → 13 taken 125 times.
✓ Branch 12 → 14 taken 1 time.
✓ Branch 15 → 16 taken 125 times.
✓ Branch 15 → 21 taken 332 times.
457 if (trueNode->hasCompileTimeValue() && falseNode->hasCompileTimeValue()) {
105 // If both are constants, we can simply emit a selection instruction
106 125 llvm::Value *trueValue = resolveValue(trueNode);
107 125 llvm::Value *falseValue = resolveValue(falseNode);
108
2/4
✓ Branch 18 → 19 taken 125 times.
✗ Branch 18 → 151 not taken.
✓ Branch 19 → 20 taken 125 times.
✗ Branch 19 → 151 not taken.
125 resultValue = builder.CreateSelect(condValue, trueValue, falseValue);
109 } else {
110 // We have at least one non-constant value, use branching to not perform both sides
111
1/2
✓ Branch 21 → 22 taken 332 times.
✗ Branch 21 → 210 not taken.
332 const std::string codeLoc = node->codeLoc.toPrettyLineAndColumn();
112
2/4
✓ Branch 22 → 23 taken 332 times.
✗ Branch 22 → 154 not taken.
✓ Branch 23 → 24 taken 332 times.
✗ Branch 23 → 152 not taken.
332 llvm::BasicBlock *condTrue = createBlock("cond.true." + codeLoc);
113
2/4
✓ Branch 25 → 26 taken 332 times.
✗ Branch 25 → 157 not taken.
✓ Branch 26 → 27 taken 332 times.
✗ Branch 26 → 155 not taken.
332 llvm::BasicBlock *condFalse = createBlock("cond.false." + codeLoc);
114
2/4
✓ Branch 28 → 29 taken 332 times.
✗ Branch 28 → 160 not taken.
✓ Branch 29 → 30 taken 332 times.
✗ Branch 29 → 158 not taken.
332 llvm::BasicBlock *condExit = createBlock("cond.exit." + codeLoc);
115
116 // Jump from original block to true or false block, depending on condition
117
1/2
✓ Branch 31 → 32 taken 332 times.
✗ Branch 31 → 208 not taken.
332 insertCondJump(condValue, condTrue, condFalse);
118
119 // Fill true block
120
1/2
✓ Branch 32 → 33 taken 332 times.
✗ Branch 32 → 208 not taken.
332 switchToBlock(condTrue);
121
1/2
✓ Branch 33 → 34 taken 332 times.
✗ Branch 33 → 208 not taken.
332 const QualType &resultType = node->getEvaluatedSymbolType(manIdx);
122 332 llvm::Value *trueValue = nullptr;
123 332 llvm::Value *truePtr = nullptr;
124
7/8
✓ Branch 34 → 35 taken 324 times.
✓ Branch 34 → 37 taken 8 times.
✓ Branch 35 → 36 taken 324 times.
✗ Branch 35 → 208 not taken.
✓ Branch 36 → 37 taken 8 times.
✓ Branch 36 → 38 taken 316 times.
✓ Branch 39 → 40 taken 16 times.
✓ Branch 39 → 42 taken 316 times.
332 if (node->falseSideCallsCopyCtor || resultType.isRef()) { // both sides or only the false side needs copy ctor call
125
1/2
✓ Branch 40 → 41 taken 16 times.
✗ Branch 40 → 208 not taken.
16 truePtr = resolveAddress(trueNode);
126
2/2
✓ Branch 42 → 43 taken 2 times.
✓ Branch 42 → 59 taken 314 times.
316 } else if (node->trueSideCallsCopyCtor) { // only true side needs copy ctor call
127
1/2
✓ Branch 43 → 44 taken 2 times.
✗ Branch 43 → 208 not taken.
2 llvm::Value *originalPtr = resolveAddress(trueNode);
128
3/6
✓ Branch 47 → 48 taken 2 times.
✗ Branch 47 → 161 not taken.
✓ Branch 48 → 49 taken 2 times.
✗ Branch 48 → 161 not taken.
✓ Branch 49 → 50 taken 2 times.
✗ Branch 49 → 161 not taken.
2 truePtr = insertAlloca(trueNode->getEvaluatedSymbolType(manIdx).toLLVMType(sourceFile));
129
2/4
✓ Branch 54 → 55 taken 2 times.
✗ Branch 54 → 169 not taken.
✓ Branch 55 → 56 taken 2 times.
✗ Branch 55 → 167 not taken.
6 generateCtorOrDtorCall(truePtr, node->calledCopyCtor, {originalPtr});
130 } else { // neither true nor false side need copy ctor call
131
1/2
✓ Branch 59 → 60 taken 314 times.
✗ Branch 59 → 208 not taken.
314 trueValue = resolveValue(trueNode);
132 }
133 // Set the true block to the current insert point, since it could have changed in the meantime
134 332 condTrue = builder.GetInsertBlock();
135
1/2
✓ Branch 62 → 63 taken 332 times.
✗ Branch 62 → 208 not taken.
332 insertJump(condExit);
136
137 // Fill false block
138
1/2
✓ Branch 63 → 64 taken 332 times.
✗ Branch 63 → 208 not taken.
332 switchToBlock(condFalse);
139 332 llvm::Value *falseValue = nullptr;
140 332 llvm::Value *falsePtr = nullptr;
141
7/8
✓ Branch 64 → 65 taken 324 times.
✓ Branch 64 → 67 taken 8 times.
✓ Branch 65 → 66 taken 324 times.
✗ Branch 65 → 208 not taken.
✓ Branch 66 → 67 taken 8 times.
✓ Branch 66 → 68 taken 316 times.
✓ Branch 69 → 70 taken 16 times.
✓ Branch 69 → 72 taken 316 times.
332 if (node->trueSideCallsCopyCtor || resultType.isRef()) { // both sides or only the true side needs copy ctor call
142
1/2
✓ Branch 70 → 71 taken 16 times.
✗ Branch 70 → 208 not taken.
16 falsePtr = resolveAddress(falseNode);
143
2/2
✓ Branch 72 → 73 taken 2 times.
✓ Branch 72 → 89 taken 314 times.
316 } else if (node->falseSideCallsCopyCtor) { // only false side needs copy ctor call
144
1/2
✓ Branch 73 → 74 taken 2 times.
✗ Branch 73 → 208 not taken.
2 llvm::Value *originalPtr = resolveAddress(falseNode);
145
3/6
✓ Branch 77 → 78 taken 2 times.
✗ Branch 77 → 174 not taken.
✓ Branch 78 → 79 taken 2 times.
✗ Branch 78 → 174 not taken.
✓ Branch 79 → 80 taken 2 times.
✗ Branch 79 → 174 not taken.
2 falsePtr = insertAlloca(falseNode->getEvaluatedSymbolType(manIdx).toLLVMType(sourceFile));
146
2/4
✓ Branch 84 → 85 taken 2 times.
✗ Branch 84 → 182 not taken.
✓ Branch 85 → 86 taken 2 times.
✗ Branch 85 → 180 not taken.
6 generateCtorOrDtorCall(falsePtr, node->calledCopyCtor, {originalPtr});
147 } else { // neither true nor false side need copy ctor call
148
1/2
✓ Branch 89 → 90 taken 314 times.
✗ Branch 89 → 208 not taken.
314 falseValue = resolveValue(falseNode);
149 }
150 // Set the true block to the current insert point, since it could have changed in the meantime
151 332 condFalse = builder.GetInsertBlock();
152
1/2
✓ Branch 92 → 93 taken 332 times.
✗ Branch 92 → 208 not taken.
332 insertJump(condExit);
153
154 // Fill the exit block
155
1/2
✓ Branch 93 → 94 taken 332 times.
✗ Branch 93 → 208 not taken.
332 switchToBlock(condExit);
156
9/10
✓ Branch 94 → 95 taken 324 times.
✓ Branch 94 → 98 taken 8 times.
✓ Branch 95 → 96 taken 322 times.
✓ Branch 95 → 98 taken 2 times.
✓ Branch 96 → 97 taken 322 times.
✗ Branch 96 → 208 not taken.
✓ Branch 97 → 98 taken 8 times.
✓ Branch 97 → 99 taken 314 times.
✓ Branch 100 → 101 taken 18 times.
✓ Branch 100 → 124 taken 314 times.
332 if (node->trueSideCallsCopyCtor || node->falseSideCallsCopyCtor || resultType.isRef()) { // one side calls copy ctor
157
3/6
✓ Branch 101 → 102 taken 18 times.
✗ Branch 101 → 187 not taken.
✓ Branch 102 → 103 taken 18 times.
✗ Branch 102 → 187 not taken.
✓ Branch 103 → 104 taken 18 times.
✗ Branch 103 → 187 not taken.
18 llvm::PHINode *phiInst = builder.CreatePHI(builder.getPtrTy(), 2, "cond.result");
158
1/2
✓ Branch 104 → 105 taken 18 times.
✗ Branch 104 → 208 not taken.
18 phiInst->addIncoming(truePtr, condTrue);
159
1/2
✓ Branch 105 → 106 taken 18 times.
✗ Branch 105 → 208 not taken.
18 phiInst->addIncoming(falsePtr, condFalse);
160
4/4
✓ Branch 106 → 107 taken 8 times.
✓ Branch 106 → 122 taken 10 times.
✓ Branch 107 → 108 taken 6 times.
✓ Branch 107 → 122 taken 2 times.
18 if (node->trueSideCallsCopyCtor && node->falseSideCallsCopyCtor) { // both sides need copy ctor call
161
2/4
✓ Branch 111 → 112 taken 6 times.
✗ Branch 111 → 188 not taken.
✓ Branch 112 → 113 taken 6 times.
✗ Branch 112 → 188 not taken.
6 resultPtr = insertAlloca(resultType.toLLVMType(sourceFile));
162
2/4
✓ Branch 117 → 118 taken 6 times.
✗ Branch 117 → 196 not taken.
✓ Branch 118 → 119 taken 6 times.
✗ Branch 118 → 194 not taken.
18 generateCtorOrDtorCall(resultPtr, node->calledCopyCtor, {phiInst});
163 } else {
164 12 resultPtr = phiInst;
165 }
166 } else { // neither true nor false side calls copy ctor
167
1/2
✗ Branch 124 → 125 not taken.
✓ Branch 124 → 126 taken 314 times.
314 assert(trueValue != nullptr);
168
3/6
✓ Branch 126 → 127 taken 314 times.
✗ Branch 126 → 201 not taken.
✓ Branch 127 → 128 taken 314 times.
✗ Branch 127 → 201 not taken.
✓ Branch 128 → 129 taken 314 times.
✗ Branch 128 → 201 not taken.
314 llvm::PHINode *phiInst = builder.CreatePHI(resultType.toLLVMType(sourceFile), 2, "cond.result");
169
1/2
✓ Branch 129 → 130 taken 314 times.
✗ Branch 129 → 208 not taken.
314 phiInst->addIncoming(trueValue, condTrue);
170
1/2
✓ Branch 130 → 131 taken 314 times.
✗ Branch 130 → 208 not taken.
314 phiInst->addIncoming(falseValue, condFalse);
171 314 resultValue = phiInst;
172 }
173
174 // If we have an anonymous symbol for this ternary expr, make sure that it has an address to reference.
175
1/2
✓ Branch 132 → 133 taken 332 times.
✗ Branch 132 → 208 not taken.
332 anonymousSymbol = currentScope->symbolTable.lookupAnonymous(node->codeLoc);
176
2/2
✓ Branch 133 → 134 taken 6 times.
✓ Branch 133 → 145 taken 326 times.
332 if (anonymousSymbol != nullptr) {
177
1/2
✗ Branch 134 → 135 not taken.
✓ Branch 134 → 144 taken 6 times.
6 if (!resultPtr) {
178 resultPtr = insertAlloca(anonymousSymbol->getQualType().toLLVMType(sourceFile));
179 insertStore(resultValue, resultPtr);
180 }
181
1/2
✓ Branch 144 → 145 taken 6 times.
✗ Branch 144 → 208 not taken.
6 anonymousSymbol->updateAddress(resultPtr);
182 }
183 332 }
184
185
1/2
✓ Branch 147 → 148 taken 457 times.
✗ Branch 147 → 211 not taken.
914 return LLVMExprResult{.value = resultValue, .ptr = resultPtr, .entry = anonymousSymbol};
186 }
187
188 61086 std::any IRGenerator::visitLogicalOrExpr(const LogicalOrExprNode *node) {
189
1/2
✓ Branch 2 → 3 taken 61086 times.
✗ Branch 2 → 96 not taken.
61086 diGenerator.setSourceLocation(node);
190
191 // Check if only one operand is present -> loop through
192
2/2
✓ Branch 4 → 5 taken 60211 times.
✓ Branch 4 → 8 taken 875 times.
61086 if (node->operands.size() == 1)
193
1/2
✓ Branch 6 → 7 taken 60211 times.
✗ Branch 6 → 96 not taken.
60211 return visit(node->operands.front());
194
195 // It is a logical or expression
196 // Create exit block for short-circuiting
197
1/2
✓ Branch 8 → 9 taken 875 times.
✗ Branch 8 → 96 not taken.
875 const std::string codeLoc = node->codeLoc.toPrettyLineAndColumn();
198
2/4
✓ Branch 9 → 10 taken 875 times.
✗ Branch 9 → 73 not taken.
✓ Branch 10 → 11 taken 875 times.
✗ Branch 10 → 71 not taken.
875 llvm::BasicBlock *bExit = createBlock("lor.exit." + codeLoc);
199
200 // Visit the first operand
201
1/2
✓ Branch 13 → 14 taken 875 times.
✗ Branch 13 → 94 not taken.
875 llvm::Value *firstOperandValue = resolveValue(node->operands.front());
202
203 // Prepare an array for value-to-block-mapping
204 875 std::vector<std::pair<llvm::BasicBlock *, llvm::Value *>> shortCircuitBlocks;
205
1/2
✓ Branch 15 → 16 taken 875 times.
✗ Branch 15 → 92 not taken.
875 shortCircuitBlocks.reserve(node->operands.size());
206 // The first element is the first operand value with the original block
207
1/2
✓ Branch 17 → 18 taken 875 times.
✗ Branch 17 → 74 not taken.
875 shortCircuitBlocks.emplace_back(builder.GetInsertBlock(), firstOperandValue);
208 // Create a block for each additional operand and save it to the mapping
209
2/2
✓ Branch 31 → 19 taken 1124 times.
✓ Branch 31 → 32 taken 875 times.
1999 for (size_t i = 1; i < node->operands.size(); i++)
210
6/12
✓ Branch 19 → 20 taken 1124 times.
✗ Branch 19 → 83 not taken.
✓ Branch 20 → 21 taken 1124 times.
✗ Branch 20 → 81 not taken.
✓ Branch 21 → 22 taken 1124 times.
✗ Branch 21 → 79 not taken.
✓ Branch 22 → 23 taken 1124 times.
✗ Branch 22 → 77 not taken.
✓ Branch 23 → 24 taken 1124 times.
✗ Branch 23 → 75 not taken.
✓ Branch 24 → 25 taken 1124 times.
✗ Branch 24 → 75 not taken.
1124 shortCircuitBlocks.emplace_back(createBlock("lor." + std::to_string(i) + "." + codeLoc), nullptr);
211 // Create conditional jump to the exit block if the first operand was true, otherwise to the next block
212
2/4
✓ Branch 32 → 33 taken 875 times.
✗ Branch 32 → 92 not taken.
✓ Branch 33 → 34 taken 875 times.
✗ Branch 33 → 92 not taken.
875 insertCondJump(firstOperandValue, bExit, shortCircuitBlocks.at(1).first);
213
214 // Create block for each operand
215
2/2
✓ Branch 50 → 35 taken 1124 times.
✓ Branch 50 → 51 taken 875 times.
1999 for (size_t i = 1; i < node->operands.size(); i++) {
216 // Switch to the next block
217
2/4
✓ Branch 35 → 36 taken 1124 times.
✗ Branch 35 → 92 not taken.
✓ Branch 36 → 37 taken 1124 times.
✗ Branch 36 → 92 not taken.
1124 switchToBlock(shortCircuitBlocks.at(i).first);
218 // Evaluate operand and save the result in the mapping
219
2/4
✓ Branch 38 → 39 taken 1124 times.
✗ Branch 38 → 92 not taken.
✓ Branch 39 → 40 taken 1124 times.
✗ Branch 39 → 92 not taken.
1124 shortCircuitBlocks.at(i).second = resolveValue(node->operands[i]);
220 // Replace the array entry with the current insert block, since the insert block could have changed in the meantime
221
1/2
✓ Branch 41 → 42 taken 1124 times.
✗ Branch 41 → 92 not taken.
1124 shortCircuitBlocks.at(i).first = builder.GetInsertBlock();
222 // Check if there are more blocks to process
223
2/2
✓ Branch 43 → 44 taken 875 times.
✓ Branch 43 → 45 taken 249 times.
1124 if (i == node->operands.size() - 1) {
224 // Insert a simple jump to the exit block for the last block
225
1/2
✓ Branch 44 → 48 taken 875 times.
✗ Branch 44 → 92 not taken.
875 insertJump(bExit);
226 } else {
227 // Create conditional jump to the exit block if the first operand was true, otherwise to the next block
228
3/6
✓ Branch 45 → 46 taken 249 times.
✗ Branch 45 → 92 not taken.
✓ Branch 46 → 47 taken 249 times.
✗ Branch 46 → 92 not taken.
✓ Branch 47 → 48 taken 249 times.
✗ Branch 47 → 92 not taken.
249 insertCondJump(shortCircuitBlocks.at(i).second, bExit, shortCircuitBlocks.at(i + 1).first);
229 }
230 }
231
232 // Get the result with the phi node
233
1/2
✓ Branch 51 → 52 taken 875 times.
✗ Branch 51 → 92 not taken.
875 switchToBlock(bExit);
234
2/4
✓ Branch 52 → 53 taken 875 times.
✗ Branch 52 → 89 not taken.
✓ Branch 55 → 56 taken 875 times.
✗ Branch 55 → 89 not taken.
875 llvm::PHINode *result = builder.CreatePHI(firstOperandValue->getType(), node->operands.size(), "lor_phi");
235
2/2
✓ Branch 64 → 58 taken 1999 times.
✓ Branch 64 → 65 taken 875 times.
2874 for (const auto &[incomingBlock, value] : shortCircuitBlocks)
236
1/2
✓ Branch 61 → 62 taken 1999 times.
✗ Branch 61 → 90 not taken.
1999 result->addIncoming(value, incomingBlock);
237
238 // Return the result
239
1/2
✓ Branch 65 → 66 taken 875 times.
✗ Branch 65 → 91 not taken.
875 return LLVMExprResult{.value = result};
240 875 }
241
242 62210 std::any IRGenerator::visitLogicalAndExpr(const LogicalAndExprNode *node) {
243
1/2
✓ Branch 2 → 3 taken 62210 times.
✗ Branch 2 → 96 not taken.
62210 diGenerator.setSourceLocation(node);
244
245 // Check if only one operand is present -> loop through
246
2/2
✓ Branch 4 → 5 taken 62045 times.
✓ Branch 4 → 8 taken 165 times.
62210 if (node->operands.size() == 1)
247
1/2
✓ Branch 6 → 7 taken 62045 times.
✗ Branch 6 → 96 not taken.
62045 return visit(node->operands.front());
248
249 // It is a logical and expression
250 // Create exit block for short-circuiting
251
1/2
✓ Branch 8 → 9 taken 165 times.
✗ Branch 8 → 96 not taken.
165 const std::string codeLoc = node->codeLoc.toPrettyLineAndColumn();
252
2/4
✓ Branch 9 → 10 taken 165 times.
✗ Branch 9 → 73 not taken.
✓ Branch 10 → 11 taken 165 times.
✗ Branch 10 → 71 not taken.
165 llvm::BasicBlock *bExit = createBlock("land.exit." + codeLoc);
253
254 // Visit the first operand
255
1/2
✓ Branch 13 → 14 taken 165 times.
✗ Branch 13 → 94 not taken.
165 llvm::Value *firstOperandValue = resolveValue(node->operands.front());
256
257 // Prepare an array for value-to-block-mapping
258 165 std::vector<std::pair<llvm::BasicBlock *, llvm::Value *>> shortCircuitBlocks;
259
1/2
✓ Branch 15 → 16 taken 165 times.
✗ Branch 15 → 92 not taken.
165 shortCircuitBlocks.reserve(node->operands.size());
260 // The first element is the first operand value with the original block
261
1/2
✓ Branch 17 → 18 taken 165 times.
✗ Branch 17 → 74 not taken.
165 shortCircuitBlocks.emplace_back(builder.GetInsertBlock(), firstOperandValue);
262 // Create a block for each additional operand and save it to the mapping
263
2/2
✓ Branch 31 → 19 taken 198 times.
✓ Branch 31 → 32 taken 165 times.
363 for (size_t i = 1; i < node->operands.size(); i++)
264
6/12
✓ Branch 19 → 20 taken 198 times.
✗ Branch 19 → 83 not taken.
✓ Branch 20 → 21 taken 198 times.
✗ Branch 20 → 81 not taken.
✓ Branch 21 → 22 taken 198 times.
✗ Branch 21 → 79 not taken.
✓ Branch 22 → 23 taken 198 times.
✗ Branch 22 → 77 not taken.
✓ Branch 23 → 24 taken 198 times.
✗ Branch 23 → 75 not taken.
✓ Branch 24 → 25 taken 198 times.
✗ Branch 24 → 75 not taken.
198 shortCircuitBlocks.emplace_back(createBlock("land." + std::to_string(i) + "." + codeLoc), nullptr);
265 // Create conditional jump to the exit block if the first operand was true, otherwise to the next block
266
2/4
✓ Branch 32 → 33 taken 165 times.
✗ Branch 32 → 92 not taken.
✓ Branch 33 → 34 taken 165 times.
✗ Branch 33 → 92 not taken.
165 insertCondJump(firstOperandValue, shortCircuitBlocks.at(1).first, bExit);
267
268 // Create block for each operand
269
2/2
✓ Branch 50 → 35 taken 198 times.
✓ Branch 50 → 51 taken 165 times.
363 for (size_t i = 1; i < node->operands.size(); i++) {
270 // Switch to the next block
271
2/4
✓ Branch 35 → 36 taken 198 times.
✗ Branch 35 → 92 not taken.
✓ Branch 36 → 37 taken 198 times.
✗ Branch 36 → 92 not taken.
198 switchToBlock(shortCircuitBlocks.at(i).first);
272 // Evaluate operand and save the result in the mapping
273
2/4
✓ Branch 38 → 39 taken 198 times.
✗ Branch 38 → 92 not taken.
✓ Branch 39 → 40 taken 198 times.
✗ Branch 39 → 92 not taken.
198 shortCircuitBlocks.at(i).second = resolveValue(node->operands[i]);
274 // Replace the array entry with the current insert block, since the insert block could have changed in the meantime
275
1/2
✓ Branch 41 → 42 taken 198 times.
✗ Branch 41 → 92 not taken.
198 shortCircuitBlocks.at(i).first = builder.GetInsertBlock();
276 // Check if there are more blocks to process
277
2/2
✓ Branch 43 → 44 taken 165 times.
✓ Branch 43 → 45 taken 33 times.
198 if (i == node->operands.size() - 1) {
278 // Insert a simple jump to the exit block for the last block
279
1/2
✓ Branch 44 → 48 taken 165 times.
✗ Branch 44 → 92 not taken.
165 insertJump(bExit);
280 } else {
281 // Create conditional jump to the exit block if the operand was true, otherwise to the next block
282
3/6
✓ Branch 45 → 46 taken 33 times.
✗ Branch 45 → 92 not taken.
✓ Branch 46 → 47 taken 33 times.
✗ Branch 46 → 92 not taken.
✓ Branch 47 → 48 taken 33 times.
✗ Branch 47 → 92 not taken.
33 insertCondJump(shortCircuitBlocks.at(i).second, shortCircuitBlocks.at(i + 1).first, bExit);
283 }
284 }
285
286 // Get the result with the phi node
287
1/2
✓ Branch 51 → 52 taken 165 times.
✗ Branch 51 → 92 not taken.
165 switchToBlock(bExit);
288
2/4
✓ Branch 52 → 53 taken 165 times.
✗ Branch 52 → 89 not taken.
✓ Branch 55 → 56 taken 165 times.
✗ Branch 55 → 89 not taken.
165 llvm::PHINode *result = builder.CreatePHI(firstOperandValue->getType(), node->operands.size(), "land_phi");
289
2/2
✓ Branch 64 → 58 taken 363 times.
✓ Branch 64 → 65 taken 165 times.
528 for (const auto &[incomingBlock, value] : shortCircuitBlocks)
290
1/2
✓ Branch 61 → 62 taken 363 times.
✗ Branch 61 → 90 not taken.
363 result->addIncoming(value, incomingBlock);
291
292 // Return the result
293
1/2
✓ Branch 65 → 66 taken 165 times.
✗ Branch 65 → 91 not taken.
165 return LLVMExprResult{.value = result};
294 165 }
295
296 62408 std::any IRGenerator::visitBitwiseOrExpr(const BitwiseOrExprNode *node) {
297
1/2
✓ Branch 2 → 3 taken 62408 times.
✗ Branch 2 → 34 not taken.
62408 diGenerator.setSourceLocation(node);
298
299 // Check if only one operand is present -> loop through
300
2/2
✓ Branch 4 → 5 taken 62332 times.
✓ Branch 4 → 8 taken 76 times.
62408 if (node->operands.size() == 1)
301
1/2
✓ Branch 6 → 7 taken 62332 times.
✗ Branch 6 → 34 not taken.
62332 return visit(node->operands.front());
302
303 // It is a bitwise or expression
304 // Evaluate first operand
305 76 const BitwiseXorExprNode *lhsNode = node->operands.front();
306
1/2
✓ Branch 9 → 10 taken 76 times.
✗ Branch 9 → 34 not taken.
76 const QualType lhsSTy = lhsNode->getEvaluatedSymbolType(manIdx);
307
2/4
✓ Branch 10 → 11 taken 76 times.
✗ Branch 10 → 29 not taken.
✓ Branch 11 → 12 taken 76 times.
✗ Branch 11 → 27 not taken.
76 auto result = std::any_cast<LLVMExprResult>(visit(lhsNode));
308
309 // Evaluate all additional operands
310
2/2
✓ Branch 22 → 14 taken 77 times.
✓ Branch 22 → 23 taken 76 times.
153 for (size_t i = 1; i < node->operands.size(); i++) {
311 // Evaluate the operand
312 77 const BitwiseXorExprNode *rhsNode = node->operands[i];
313
1/2
✓ Branch 15 → 16 taken 77 times.
✗ Branch 15 → 33 not taken.
77 const QualType rhsSTy = lhsNode->getEvaluatedSymbolType(manIdx);
314
2/4
✓ Branch 16 → 17 taken 77 times.
✗ Branch 16 → 32 not taken.
✓ Branch 17 → 18 taken 77 times.
✗ Branch 17 → 30 not taken.
77 auto rhs = std::any_cast<LLVMExprResult>(visit(rhsNode));
315
1/2
✓ Branch 19 → 20 taken 77 times.
✗ Branch 19 → 33 not taken.
77 result = conversionManager.getBitwiseOrInst(node, result, lhsSTy, rhs, rhsSTy, i - 1);
316 }
317
318 // Return result
319
1/2
✓ Branch 23 → 24 taken 76 times.
✗ Branch 23 → 34 not taken.
76 return result;
320 }
321
322 62485 std::any IRGenerator::visitBitwiseXorExpr(const BitwiseXorExprNode *node) {
323
1/2
✓ Branch 2 → 3 taken 62485 times.
✗ Branch 2 → 34 not taken.
62485 diGenerator.setSourceLocation(node);
324
325 // Check if only one operand is present -> loop through
326
2/2
✓ Branch 4 → 5 taken 62476 times.
✓ Branch 4 → 8 taken 9 times.
62485 if (node->operands.size() == 1)
327
1/2
✓ Branch 6 → 7 taken 62476 times.
✗ Branch 6 → 34 not taken.
62476 return visit(node->operands.front());
328
329 // It is a bitwise xor expression
330 // Evaluate first operand
331 9 const BitwiseAndExprNode *lhsNode = node->operands.front();
332
1/2
✓ Branch 9 → 10 taken 9 times.
✗ Branch 9 → 34 not taken.
9 const QualType lhsSTy = lhsNode->getEvaluatedSymbolType(manIdx);
333
2/4
✓ Branch 10 → 11 taken 9 times.
✗ Branch 10 → 29 not taken.
✓ Branch 11 → 12 taken 9 times.
✗ Branch 11 → 27 not taken.
9 auto result = std::any_cast<LLVMExprResult>(visit(lhsNode));
334
335 // Evaluate all additional operands
336
2/2
✓ Branch 22 → 14 taken 10 times.
✓ Branch 22 → 23 taken 9 times.
19 for (size_t i = 1; i < node->operands.size(); i++) {
337 // Evaluate the operand
338 10 const BitwiseAndExprNode *rhsNode = node->operands[i];
339
1/2
✓ Branch 15 → 16 taken 10 times.
✗ Branch 15 → 33 not taken.
10 const QualType rhsSTy = lhsNode->getEvaluatedSymbolType(manIdx);
340
2/4
✓ Branch 16 → 17 taken 10 times.
✗ Branch 16 → 32 not taken.
✓ Branch 17 → 18 taken 10 times.
✗ Branch 17 → 30 not taken.
10 auto rhs = std::any_cast<LLVMExprResult>(visit(rhsNode));
341
1/2
✓ Branch 19 → 20 taken 10 times.
✗ Branch 19 → 33 not taken.
10 result = conversionManager.getBitwiseXorInst(node, result, lhsSTy, rhs, rhsSTy);
342 }
343
344 // Return result
345
1/2
✓ Branch 23 → 24 taken 9 times.
✗ Branch 23 → 34 not taken.
9 return result;
346 }
347
348 62495 std::any IRGenerator::visitBitwiseAndExpr(const BitwiseAndExprNode *node) {
349
1/2
✓ Branch 2 → 3 taken 62495 times.
✗ Branch 2 → 34 not taken.
62495 diGenerator.setSourceLocation(node);
350
351 // Check if only one operand is present -> loop through
352
2/2
✓ Branch 4 → 5 taken 62470 times.
✓ Branch 4 → 8 taken 25 times.
62495 if (node->operands.size() == 1)
353
1/2
✓ Branch 6 → 7 taken 62470 times.
✗ Branch 6 → 34 not taken.
62470 return visit(node->operands.front());
354
355 // It is a bitwise and expression
356 // Evaluate first operand
357 25 const EqualityExprNode *lhsNode = node->operands.front();
358
1/2
✓ Branch 9 → 10 taken 25 times.
✗ Branch 9 → 34 not taken.
25 const QualType lhsSTy = lhsNode->getEvaluatedSymbolType(manIdx);
359
2/4
✓ Branch 10 → 11 taken 25 times.
✗ Branch 10 → 29 not taken.
✓ Branch 11 → 12 taken 25 times.
✗ Branch 11 → 27 not taken.
25 auto result = std::any_cast<LLVMExprResult>(visit(lhsNode));
360
361 // Evaluate all additional operands
362
2/2
✓ Branch 22 → 14 taken 26 times.
✓ Branch 22 → 23 taken 25 times.
51 for (size_t i = 1; i < node->operands.size(); i++) {
363 // Evaluate the operand
364 26 const EqualityExprNode *rhsNode = node->operands[i];
365
1/2
✓ Branch 15 → 16 taken 26 times.
✗ Branch 15 → 33 not taken.
26 const QualType rhsSTy = lhsNode->getEvaluatedSymbolType(manIdx);
366
2/4
✓ Branch 16 → 17 taken 26 times.
✗ Branch 16 → 32 not taken.
✓ Branch 17 → 18 taken 26 times.
✗ Branch 17 → 30 not taken.
26 auto rhs = std::any_cast<LLVMExprResult>(visit(rhsNode));
367
1/2
✓ Branch 19 → 20 taken 26 times.
✗ Branch 19 → 33 not taken.
26 result = conversionManager.getBitwiseAndInst(rhsNode, result, lhsSTy, rhs, rhsSTy, i - 1);
368 }
369
370 // Return result
371
1/2
✓ Branch 23 → 24 taken 25 times.
✗ Branch 23 → 34 not taken.
25 return result;
372 }
373
374 62521 std::any IRGenerator::visitEqualityExpr(const EqualityExprNode *node) {
375
1/2
✓ Branch 2 → 3 taken 62521 times.
✗ Branch 2 → 50 not taken.
62521 diGenerator.setSourceLocation(node);
376
377 // Check if only one operand is present -> loop through
378
2/2
✓ Branch 4 → 5 taken 57697 times.
✓ Branch 4 → 8 taken 4824 times.
62521 if (node->operands.size() == 1)
379
1/2
✓ Branch 6 → 7 taken 57697 times.
✗ Branch 6 → 50 not taken.
57697 return visit(node->operands.front());
380
381 // It is an equality expression
382 // Evaluate lhs
383 4824 const RelationalExprNode *lhsNode = node->operands[0];
384
1/2
✓ Branch 9 → 10 taken 4824 times.
✗ Branch 9 → 50 not taken.
4824 const QualType lhsSTy = lhsNode->getEvaluatedSymbolType(manIdx);
385
2/4
✓ Branch 10 → 11 taken 4824 times.
✗ Branch 10 → 37 not taken.
✓ Branch 11 → 12 taken 4824 times.
✗ Branch 11 → 35 not taken.
4824 auto result = std::any_cast<LLVMExprResult>(visit(lhsNode));
386
387 // Evaluate rhs
388 4824 const RelationalExprNode *rhsNode = node->operands[1];
389
1/2
✓ Branch 14 → 15 taken 4824 times.
✗ Branch 14 → 50 not taken.
4824 const QualType rhsSTy = rhsNode->getEvaluatedSymbolType(manIdx);
390
2/4
✓ Branch 15 → 16 taken 4824 times.
✗ Branch 15 → 40 not taken.
✓ Branch 16 → 17 taken 4824 times.
✗ Branch 16 → 38 not taken.
4824 auto rhs = std::any_cast<LLVMExprResult>(visit(rhsNode));
391
392 // Retrieve the result value, based on the exact operator
393
2/3
✓ Branch 18 → 19 taken 3446 times.
✓ Branch 18 → 21 taken 1378 times.
✗ Branch 18 → 23 not taken.
4824 switch (node->op) {
394 3446 case EqualityExprNode::EqualityOp::OP_EQUAL:
395
1/2
✓ Branch 19 → 20 taken 3446 times.
✗ Branch 19 → 50 not taken.
3446 result = conversionManager.getEqualInst(node, result, lhsSTy, rhs, rhsSTy, 0);
396 3446 break;
397 1378 case EqualityExprNode::EqualityOp::OP_NOT_EQUAL:
398
1/2
✓ Branch 21 → 22 taken 1378 times.
✗ Branch 21 → 50 not taken.
1378 result = conversionManager.getNotEqualInst(node, result, lhsSTy, rhs, rhsSTy, 0);
399 1378 break;
400 default: // GCOV_EXCL_LINE
401 throw CompilerError(UNHANDLED_BRANCH, "EqualityExpr fall-through"); // GCOV_EXCL_LINE
402 }
403
404 // Return the result
405
1/2
✓ Branch 31 → 32 taken 4824 times.
✗ Branch 31 → 50 not taken.
4824 return result;
406 }
407
408 67345 std::any IRGenerator::visitRelationalExpr(const RelationalExprNode *node) {
409
1/2
✓ Branch 2 → 3 taken 67345 times.
✗ Branch 2 → 54 not taken.
67345 diGenerator.setSourceLocation(node);
410
411 // Check if only one operand is present -> loop through
412
2/2
✓ Branch 4 → 5 taken 63968 times.
✓ Branch 4 → 8 taken 3377 times.
67345 if (node->operands.size() == 1)
413
1/2
✓ Branch 6 → 7 taken 63968 times.
✗ Branch 6 → 54 not taken.
63968 return visit(node->operands.front());
414
415 // It is a relational expression
416 // Evaluate lhs
417 3377 const ShiftExprNode *lhsNode = node->operands[0];
418
1/2
✓ Branch 9 → 10 taken 3377 times.
✗ Branch 9 → 54 not taken.
3377 const QualType lhsSTy = lhsNode->getEvaluatedSymbolType(manIdx);
419
2/4
✓ Branch 10 → 11 taken 3377 times.
✗ Branch 10 → 41 not taken.
✓ Branch 11 → 12 taken 3377 times.
✗ Branch 11 → 39 not taken.
3377 auto result = std::any_cast<LLVMExprResult>(visit(lhsNode));
420
421 // Evaluate rhs
422 3377 const ShiftExprNode *rhsNode = node->operands[1];
423
1/2
✓ Branch 14 → 15 taken 3377 times.
✗ Branch 14 → 54 not taken.
3377 const QualType rhsSTy = rhsNode->getEvaluatedSymbolType(manIdx);
424
2/4
✓ Branch 15 → 16 taken 3377 times.
✗ Branch 15 → 44 not taken.
✓ Branch 16 → 17 taken 3377 times.
✗ Branch 16 → 42 not taken.
3377 auto rhs = std::any_cast<LLVMExprResult>(visit(rhsNode));
425
426 // Retrieve the result value, based on the exact operator
427
4/5
✓ Branch 18 → 19 taken 1652 times.
✓ Branch 18 → 21 taken 473 times.
✓ Branch 18 → 23 taken 346 times.
✓ Branch 18 → 25 taken 906 times.
✗ Branch 18 → 27 not taken.
3377 switch (node->op) {
428 1652 case RelationalExprNode::RelationalOp::OP_LESS:
429
1/2
✓ Branch 19 → 20 taken 1652 times.
✗ Branch 19 → 54 not taken.
1652 result = conversionManager.getLessInst(node, result, lhsSTy, rhs, rhsSTy);
430 1652 break;
431 473 case RelationalExprNode::RelationalOp::OP_GREATER:
432
1/2
✓ Branch 21 → 22 taken 473 times.
✗ Branch 21 → 54 not taken.
473 result = conversionManager.getGreaterInst(node, result, lhsSTy, rhs, rhsSTy);
433 473 break;
434 346 case RelationalExprNode::RelationalOp::OP_LESS_EQUAL:
435
1/2
✓ Branch 23 → 24 taken 346 times.
✗ Branch 23 → 54 not taken.
346 result = conversionManager.getLessEqualInst(node, result, lhsSTy, rhs, rhsSTy);
436 346 break;
437 906 case RelationalExprNode::RelationalOp::OP_GREATER_EQUAL:
438
1/2
✓ Branch 25 → 26 taken 906 times.
✗ Branch 25 → 54 not taken.
906 result = conversionManager.getGreaterEqualInst(node, result, lhsSTy, rhs, rhsSTy);
439 906 break;
440 default: // GCOV_EXCL_LINE
441 throw CompilerError(UNHANDLED_BRANCH, "EqualityExpr fall-through"); // GCOV_EXCL_LINE
442 }
443
444 // Return the result
445
1/2
✓ Branch 35 → 36 taken 3377 times.
✗ Branch 35 → 54 not taken.
3377 return result;
446 }
447
448 70722 std::any IRGenerator::visitShiftExpr(const ShiftExprNode *node) {
449
1/2
✓ Branch 2 → 3 taken 70722 times.
✗ Branch 2 → 64 not taken.
70722 diGenerator.setSourceLocation(node);
450
451 // Check if only one operand is present -> loop through
452
2/2
✓ Branch 4 → 5 taken 70611 times.
✓ Branch 4 → 8 taken 111 times.
70722 if (node->operands.size() == 1)
453
1/2
✓ Branch 6 → 7 taken 70611 times.
✗ Branch 6 → 64 not taken.
70611 return visit(node->operands.front());
454
455 // It is a shift expression
456 // Evaluate first operand
457 111 const AdditiveExprNode *lhsNode = node->operands.front();
458
1/2
✓ Branch 9 → 10 taken 111 times.
✗ Branch 9 → 64 not taken.
111 QualType lhsSTy = lhsNode->getEvaluatedSymbolType(manIdx);
459
2/4
✓ Branch 10 → 11 taken 111 times.
✗ Branch 10 → 48 not taken.
✓ Branch 11 → 12 taken 111 times.
✗ Branch 11 → 46 not taken.
111 auto lhs = std::any_cast<LLVMExprResult>(visit(lhsNode));
460
461
1/2
✓ Branch 13 → 14 taken 111 times.
✗ Branch 13 → 64 not taken.
111 auto opQueue = node->opQueue;
462 111 size_t operandIndex = 1;
463
2/2
✓ Branch 40 → 15 taken 151 times.
✓ Branch 40 → 41 taken 111 times.
262 while (!opQueue.empty()) {
464 151 const size_t operatorIndex = operandIndex - 1;
465 // Evaluate next operand
466 151 const AdditiveExprNode *rhsNode = node->operands[operandIndex++];
467
1/2
✗ Branch 16 → 17 not taken.
✓ Branch 16 → 18 taken 151 times.
151 assert(rhsNode != nullptr);
468
1/2
✓ Branch 18 → 19 taken 151 times.
✗ Branch 18 → 61 not taken.
151 const QualType rhsSTy = rhsNode->getEvaluatedSymbolType(manIdx);
469
2/4
✓ Branch 19 → 20 taken 151 times.
✗ Branch 19 → 51 not taken.
✓ Branch 20 → 21 taken 151 times.
✗ Branch 20 → 49 not taken.
151 auto rhs = std::any_cast<LLVMExprResult>(visit(rhsNode));
470
471 // Retrieve the result, based on the exact operator
472
2/3
✓ Branch 23 → 24 taken 97 times.
✓ Branch 23 → 26 taken 54 times.
✗ Branch 23 → 28 not taken.
151 switch (opQueue.front().first) {
473 97 case ShiftExprNode::ShiftOp::OP_SHIFT_LEFT:
474
1/2
✓ Branch 24 → 25 taken 97 times.
✗ Branch 24 → 61 not taken.
97 lhs = conversionManager.getShiftLeftInst(node, lhs, lhsSTy, rhs, rhsSTy, operatorIndex);
475 97 break;
476 54 case ShiftExprNode::ShiftOp::OP_SHIFT_RIGHT:
477
1/2
✓ Branch 26 → 27 taken 54 times.
✗ Branch 26 → 61 not taken.
54 lhs = conversionManager.getShiftRightInst(node, lhs, lhsSTy, rhs, rhsSTy, operatorIndex);
478 54 break;
479 default: // GCOV_EXCL_LINE
480 throw CompilerError(UNHANDLED_BRANCH, "AdditiveExpr fall-through"); // GCOV_EXCL_LINE
481 }
482
483 // Retrieve the new lhs symbol type
484 151 lhsSTy = opQueue.front().second;
485
486 151 opQueue.pop();
487 }
488
489 // Return the result
490
1/2
✓ Branch 41 → 42 taken 111 times.
✗ Branch 41 → 62 not taken.
111 return lhs;
491 111 }
492
493 70873 std::any IRGenerator::visitAdditiveExpr(const AdditiveExprNode *node) {
494
1/2
✓ Branch 2 → 3 taken 70873 times.
✗ Branch 2 → 64 not taken.
70873 diGenerator.setSourceLocation(node);
495
496 // Check if only one operand is present -> loop through
497
2/2
✓ Branch 4 → 5 taken 67074 times.
✓ Branch 4 → 8 taken 3799 times.
70873 if (node->operands.size() == 1)
498
1/2
✓ Branch 6 → 7 taken 67074 times.
✗ Branch 6 → 64 not taken.
67074 return visit(node->operands.front());
499
500 // It is an additive expression
501 // Evaluate first operand
502 3799 const MultiplicativeExprNode *lhsNode = node->operands[0];
503
1/2
✓ Branch 9 → 10 taken 3799 times.
✗ Branch 9 → 64 not taken.
3799 QualType lhsSTy = lhsNode->getEvaluatedSymbolType(manIdx);
504
2/4
✓ Branch 10 → 11 taken 3799 times.
✗ Branch 10 → 48 not taken.
✓ Branch 11 → 12 taken 3799 times.
✗ Branch 11 → 46 not taken.
3799 auto lhs = std::any_cast<LLVMExprResult>(visit(lhsNode));
505
506
1/2
✓ Branch 13 → 14 taken 3799 times.
✗ Branch 13 → 64 not taken.
3799 auto opQueue = node->opQueue;
507 3799 size_t operandIndex = 1;
508
2/2
✓ Branch 40 → 15 taken 4326 times.
✓ Branch 40 → 41 taken 3799 times.
8125 while (!opQueue.empty()) {
509 4326 const size_t operatorIndex = operandIndex - 1;
510 // Evaluate next operand
511 4326 const MultiplicativeExprNode *rhsNode = node->operands[operandIndex++];
512
1/2
✗ Branch 16 → 17 not taken.
✓ Branch 16 → 18 taken 4326 times.
4326 assert(rhsNode != nullptr);
513
1/2
✓ Branch 18 → 19 taken 4326 times.
✗ Branch 18 → 61 not taken.
4326 const QualType rhsSTy = rhsNode->getEvaluatedSymbolType(manIdx);
514
2/4
✓ Branch 19 → 20 taken 4326 times.
✗ Branch 19 → 51 not taken.
✓ Branch 20 → 21 taken 4326 times.
✗ Branch 20 → 49 not taken.
4326 auto rhs = std::any_cast<LLVMExprResult>(visit(rhsNode));
515
516 // Retrieve the result, based on the exact operator
517
2/3
✓ Branch 23 → 24 taken 2653 times.
✓ Branch 23 → 26 taken 1673 times.
✗ Branch 23 → 28 not taken.
4326 switch (opQueue.front().first) {
518 2653 case AdditiveExprNode::AdditiveOp::OP_PLUS:
519
1/2
✓ Branch 24 → 25 taken 2653 times.
✗ Branch 24 → 61 not taken.
2653 lhs = conversionManager.getPlusInst(node, lhs, lhsSTy, rhs, rhsSTy, operatorIndex);
520 2653 break;
521 1673 case AdditiveExprNode::AdditiveOp::OP_MINUS:
522
1/2
✓ Branch 26 → 27 taken 1673 times.
✗ Branch 26 → 61 not taken.
1673 lhs = conversionManager.getMinusInst(node, lhs, lhsSTy, rhs, rhsSTy, operatorIndex);
523 1673 break;
524 default: // GCOV_EXCL_LINE
525 throw CompilerError(UNHANDLED_BRANCH, "AdditiveExpr fall-through"); // GCOV_EXCL_LINE
526 }
527
528 // Retrieve the new lhs symbol type
529 4326 lhsSTy = opQueue.front().second;
530
531 4326 opQueue.pop();
532 }
533
534 // Return the result
535
1/2
✓ Branch 41 → 42 taken 3799 times.
✗ Branch 41 → 62 not taken.
3799 return lhs;
536 3799 }
537
538 75199 std::any IRGenerator::visitMultiplicativeExpr(const MultiplicativeExprNode *node) {
539
1/2
✓ Branch 2 → 3 taken 75199 times.
✗ Branch 2 → 66 not taken.
75199 diGenerator.setSourceLocation(node);
540
541 // Check if only one operand is present -> loop through
542
2/2
✓ Branch 4 → 5 taken 74391 times.
✓ Branch 4 → 8 taken 808 times.
75199 if (node->operands.size() == 1)
543
1/2
✓ Branch 6 → 7 taken 74391 times.
✗ Branch 6 → 66 not taken.
74391 return visit(node->operands.front());
544
545 // It is an additive expression
546 // Evaluate first operand
547 808 const CastExprNode *lhsNode = node->operands[0];
548
1/2
✓ Branch 9 → 10 taken 808 times.
✗ Branch 9 → 66 not taken.
808 QualType lhsSTy = lhsNode->getEvaluatedSymbolType(manIdx);
549
2/4
✓ Branch 10 → 11 taken 808 times.
✗ Branch 10 → 50 not taken.
✓ Branch 11 → 12 taken 808 times.
✗ Branch 11 → 48 not taken.
808 auto result = std::any_cast<LLVMExprResult>(visit(lhsNode));
550
551
1/2
✓ Branch 13 → 14 taken 808 times.
✗ Branch 13 → 66 not taken.
808 auto opQueue = node->opQueue;
552 808 size_t operandIndex = 1;
553
2/2
✓ Branch 42 → 15 taken 828 times.
✓ Branch 42 → 43 taken 808 times.
1636 while (!opQueue.empty()) {
554 828 const size_t operatorIndex = operandIndex - 1;
555 // Evaluate next operand
556 828 const CastExprNode *rhsNode = node->operands[operandIndex++];
557
1/2
✗ Branch 16 → 17 not taken.
✓ Branch 16 → 18 taken 828 times.
828 assert(rhsNode != nullptr);
558
1/2
✓ Branch 18 → 19 taken 828 times.
✗ Branch 18 → 63 not taken.
828 const QualType rhsSTy = rhsNode->getEvaluatedSymbolType(manIdx);
559
2/4
✓ Branch 19 → 20 taken 828 times.
✗ Branch 19 → 53 not taken.
✓ Branch 20 → 21 taken 828 times.
✗ Branch 20 → 51 not taken.
828 auto rhs = std::any_cast<LLVMExprResult>(visit(rhsNode));
560
561 // Retrieve the result, based on the exact operator
562
3/4
✓ Branch 23 → 24 taken 690 times.
✓ Branch 23 → 26 taken 126 times.
✓ Branch 23 → 28 taken 12 times.
✗ Branch 23 → 30 not taken.
828 switch (opQueue.front().first) {
563 690 case MultiplicativeExprNode::MultiplicativeOp::OP_MUL:
564
1/2
✓ Branch 24 → 25 taken 690 times.
✗ Branch 24 → 63 not taken.
690 result = conversionManager.getMulInst(node, result, lhsSTy, rhs, rhsSTy, operatorIndex);
565 690 break;
566 126 case MultiplicativeExprNode::MultiplicativeOp::OP_DIV:
567
1/2
✓ Branch 26 → 27 taken 126 times.
✗ Branch 26 → 63 not taken.
126 result = conversionManager.getDivInst(node, result, lhsSTy, rhs, rhsSTy, operatorIndex);
568 126 break;
569 12 case MultiplicativeExprNode::MultiplicativeOp::OP_REM:
570
1/2
✓ Branch 28 → 29 taken 12 times.
✗ Branch 28 → 63 not taken.
12 result = conversionManager.getRemInst(node, result, lhsSTy, rhs, rhsSTy);
571 12 break;
572 default: // GCOV_EXCL_LINE
573 throw CompilerError(UNHANDLED_BRANCH, "MultiplicativeExpr fall-through"); // GCOV_EXCL_LINE
574 }
575
576 // Retrieve the new lhs symbol type
577 828 lhsSTy = opQueue.front().second;
578 828 opQueue.pop();
579 }
580
581 // Return the result
582
1/2
✓ Branch 43 → 44 taken 808 times.
✗ Branch 43 → 64 not taken.
808 return result;
583 808 }
584
585 76027 std::any IRGenerator::visitCastExpr(const CastExprNode *node) {
586
1/2
✓ Branch 2 → 3 taken 76027 times.
✗ Branch 2 → 19 not taken.
76027 diGenerator.setSourceLocation(node);
587
588 // Check if only one operand is present -> loop through
589
2/2
✓ Branch 3 → 4 taken 73687 times.
✓ Branch 3 → 6 taken 2340 times.
76027 if (!node->isCast)
590
1/2
✓ Branch 4 → 5 taken 73687 times.
✗ Branch 4 → 19 not taken.
73687 return visit(node->prefixUnaryExpr);
591
592 // It is a cast expression
593 // Retrieve target symbol type
594
1/2
✓ Branch 6 → 7 taken 2340 times.
✗ Branch 6 → 19 not taken.
2340 const QualType targetSTy = node->getEvaluatedSymbolType(manIdx);
595
596 // Evaluate rhs
597 2340 const AssignExprNode *rhsNode = node->assignExpr;
598
1/2
✓ Branch 7 → 8 taken 2340 times.
✗ Branch 7 → 19 not taken.
2340 const QualType rhsSTy = rhsNode->getEvaluatedSymbolType(manIdx);
599
2/4
✓ Branch 8 → 9 taken 2340 times.
✗ Branch 8 → 18 not taken.
✓ Branch 9 → 10 taken 2340 times.
✗ Branch 9 → 16 not taken.
2340 auto rhs = std::any_cast<LLVMExprResult>(visit(rhsNode));
600
601 // Retrieve the result value
602
1/2
✓ Branch 11 → 12 taken 2340 times.
✗ Branch 11 → 19 not taken.
2340 const LLVMExprResult result = conversionManager.getCastInst(node, targetSTy, rhs, rhsSTy);
603
604 // Return the result
605
1/2
✓ Branch 12 → 13 taken 2340 times.
✗ Branch 12 → 19 not taken.
2340 return result;
606 }
607
608 80647 std::any IRGenerator::visitPrefixUnaryExpr(const PrefixUnaryExprNode *node) {
609
1/2
✓ Branch 2 → 3 taken 80647 times.
✗ Branch 2 → 107 not taken.
80647 diGenerator.setSourceLocation(node);
610
611 // If no operator is applied, simply visit the atomic expression
612
2/2
✓ Branch 3 → 4 taken 79637 times.
✓ Branch 3 → 6 taken 1010 times.
80647 if (node->op == PrefixUnaryExprNode::PrefixUnaryOp::OP_NONE)
613
1/2
✓ Branch 4 → 5 taken 79637 times.
✗ Branch 4 → 107 not taken.
79637 return visit(node->postfixUnaryExpr);
614
615 // Evaluate lhs
616 1010 const PrefixUnaryExprNode *lhsNode = node->prefixUnaryExpr;
617
1/2
✓ Branch 6 → 7 taken 1010 times.
✗ Branch 6 → 107 not taken.
1010 const QualType lhsSTy = lhsNode->getEvaluatedSymbolType(manIdx);
618
2/4
✓ Branch 7 → 8 taken 1010 times.
✗ Branch 7 → 77 not taken.
✓ Branch 8 → 9 taken 1010 times.
✗ Branch 8 → 75 not taken.
1010 auto lhs = std::any_cast<LLVMExprResult>(visit(lhsNode));
619
620
7/8
✓ Branch 10 → 11 taken 16 times.
✓ Branch 10 → 21 taken 18 times.
✓ Branch 10 → 37 taken 4 times.
✓ Branch 10 → 53 taken 690 times.
✓ Branch 10 → 56 taken 1 time.
✓ Branch 10 → 59 taken 197 times.
✓ Branch 10 → 61 taken 84 times.
✗ Branch 10 → 63 not taken.
1010 switch (node->op) {
621 16 case PrefixUnaryExprNode::PrefixUnaryOp::OP_MINUS: {
622 // Execute operation
623
1/2
✓ Branch 11 → 12 taken 16 times.
✗ Branch 11 → 107 not taken.
16 lhs = conversionManager.getPrefixMinusInst(node, lhs, lhsSTy);
624
625 // This operator can not work in-place, so we need additional memory
626
1/2
✓ Branch 16 → 17 taken 16 times.
✗ Branch 16 → 78 not taken.
16 lhs.ptr = insertAlloca(lhs.value->getType());
627
628 // Store the new value
629
1/2
✓ Branch 19 → 20 taken 16 times.
✗ Branch 19 → 107 not taken.
16 insertStore(lhs.value, lhs.ptr);
630
631 16 break;
632 }
633 18 case PrefixUnaryExprNode::PrefixUnaryOp::OP_PLUS_PLUS: {
634 // Make sure the value is present
635
1/2
✓ Branch 21 → 22 taken 18 times.
✗ Branch 21 → 107 not taken.
18 resolveValue(lhsNode, lhs);
636
637 // Execute operation
638
1/2
✓ Branch 22 → 23 taken 18 times.
✗ Branch 22 → 84 not taken.
18 lhs.value = conversionManager.getPrefixPlusPlusInst(node, lhs, lhsSTy).value;
639
640 // If this operation happens on a volatile variable, store the value directly
641
3/4
✓ Branch 23 → 24 taken 17 times.
✓ Branch 23 → 26 taken 1 time.
✗ Branch 24 → 25 not taken.
✓ Branch 24 → 26 taken 17 times.
18 if (lhs.entry && lhs.entry->isVolatile)
642 insertStore(lhs.value, lhs.ptr, true);
643
644 // Save to the existing address if possible, otherwise (e.g. for literals) allocate new space
645
2/2
✓ Branch 26 → 27 taken 1 time.
✓ Branch 26 → 35 taken 17 times.
18 if (!lhs.ptr)
646
1/2
✓ Branch 31 → 32 taken 1 time.
✗ Branch 31 → 85 not taken.
1 lhs.ptr = insertAlloca(lhs.value->getType());
647
648 // Store the new value
649
1/2
✓ Branch 35 → 36 taken 18 times.
✗ Branch 35 → 107 not taken.
18 insertStore(lhs.value, lhs.ptr);
650
651 18 break;
652 }
653 4 case PrefixUnaryExprNode::PrefixUnaryOp::OP_MINUS_MINUS: {
654 // Make sure the value is present
655
1/2
✓ Branch 37 → 38 taken 4 times.
✗ Branch 37 → 107 not taken.
4 resolveValue(lhsNode, lhs);
656
657 // Execute operation
658
1/2
✓ Branch 38 → 39 taken 4 times.
✗ Branch 38 → 91 not taken.
4 lhs.value = conversionManager.getPrefixMinusMinusInst(node, lhs, lhsSTy).value;
659
660 // If this operation happens on a volatile variable, store the value directly
661
3/4
✓ Branch 39 → 40 taken 3 times.
✓ Branch 39 → 42 taken 1 time.
✗ Branch 40 → 41 not taken.
✓ Branch 40 → 42 taken 3 times.
4 if (lhs.entry && lhs.entry->isVolatile)
662 insertStore(lhs.value, lhs.ptr, true);
663
664 // Save to the existing address if possible, otherwise (e.g. for literals) allocate new space
665
2/2
✓ Branch 42 → 43 taken 1 time.
✓ Branch 42 → 51 taken 3 times.
4 if (!lhs.ptr)
666
1/2
✓ Branch 47 → 48 taken 1 time.
✗ Branch 47 → 92 not taken.
1 lhs.ptr = insertAlloca(lhs.value->getType());
667
668 // Store the new value
669
1/2
✓ Branch 51 → 52 taken 4 times.
✗ Branch 51 → 107 not taken.
4 insertStore(lhs.value, lhs.ptr);
670
671 4 break;
672 }
673 690 case PrefixUnaryExprNode::PrefixUnaryOp::OP_NOT: {
674 // Make sure the value is present
675
1/2
✓ Branch 53 → 54 taken 690 times.
✗ Branch 53 → 107 not taken.
690 resolveValue(lhsNode, lhs);
676
677 // Execute operation
678
1/2
✓ Branch 54 → 55 taken 690 times.
✗ Branch 54 → 107 not taken.
690 lhs = conversionManager.getPrefixNotInst(node, lhs, lhsSTy);
679
680 690 break;
681 }
682 1 case PrefixUnaryExprNode::PrefixUnaryOp::OP_BITWISE_NOT: {
683 // Make sure the value is present
684
1/2
✓ Branch 56 → 57 taken 1 time.
✗ Branch 56 → 107 not taken.
1 resolveValue(lhsNode, lhs);
685
686 // Execute operation
687
1/2
✓ Branch 57 → 58 taken 1 time.
✗ Branch 57 → 107 not taken.
1 lhs = conversionManager.getPrefixBitwiseNotInst(node, lhs, lhsSTy);
688
689 1 break;
690 }
691 197 case PrefixUnaryExprNode::PrefixUnaryOp::OP_DEREFERENCE: {
692 // Make sure the value is present
693
1/2
✓ Branch 59 → 60 taken 197 times.
✗ Branch 59 → 107 not taken.
197 resolveValue(lhsNode, lhs);
694
695 // Execute operation
696 197 lhs.ptr = lhs.value;
697
698 // Reset the value
699 197 lhs.value = nullptr;
700
701 197 break;
702 }
703 84 case PrefixUnaryExprNode::PrefixUnaryOp::OP_ADDRESS_OF: {
704 // Make sure the address is present
705
1/2
✓ Branch 61 → 62 taken 84 times.
✗ Branch 61 → 107 not taken.
84 resolveAddress(lhs);
706
707 // Execute operation
708 84 lhs.value = lhs.ptr;
709
710 // Reset the address
711 84 lhs.ptr = nullptr;
712
713 84 break;
714 }
715 default: // GCOV_EXCL_LINE
716 throw CompilerError(UNHANDLED_BRANCH, "PrefixUnaryExpr fall-through"); // GCOV_EXCL_LINE
717 }
718
719
1/2
✓ Branch 71 → 72 taken 1010 times.
✗ Branch 71 → 107 not taken.
1010 return lhs;
720 }
721
722 99399 std::any IRGenerator::visitPostfixUnaryExpr(const PostfixUnaryExprNode *node) {
723
1/2
✓ Branch 2 → 3 taken 99399 times.
✗ Branch 2 → 243 not taken.
99399 diGenerator.setSourceLocation(node);
724
725 // If no operator is applied, simply visit the atomic expression
726
2/2
✓ Branch 3 → 4 taken 79637 times.
✓ Branch 3 → 6 taken 19762 times.
99399 if (node->op == PostfixUnaryExprNode::PostfixUnaryOp::OP_NONE)
727
1/2
✓ Branch 4 → 5 taken 79637 times.
✗ Branch 4 → 243 not taken.
79637 return visit(node->atomicExpr);
728
729 // Evaluate lhs
730 19762 const PostfixUnaryExprNode *lhsNode = node->postfixUnaryExpr;
731
1/2
✓ Branch 6 → 7 taken 19762 times.
✗ Branch 6 → 243 not taken.
19762 QualType lhsSTy = lhsNode->getEvaluatedSymbolType(manIdx);
732
2/4
✓ Branch 7 → 8 taken 19762 times.
✗ Branch 7 → 165 not taken.
✓ Branch 8 → 9 taken 19762 times.
✗ Branch 8 → 163 not taken.
19762 auto lhs = std::any_cast<LLVMExprResult>(visit(lhsNode));
733
734
4/5
✓ Branch 10 → 11 taken 2983 times.
✓ Branch 10 → 64 taken 14814 times.
✓ Branch 10 → 103 taken 1603 times.
✓ Branch 10 → 127 taken 362 times.
✗ Branch 10 → 151 not taken.
19762 switch (node->op) {
735 2983 case PostfixUnaryExprNode::PostfixUnaryOp::OP_SUBSCRIPT: {
736 2983 const AssignExprNode *indexExpr = node->subscriptIndexExpr;
737
738 // Check if we need to generate a call to an overloaded operator function
739
3/4
✓ Branch 11 → 12 taken 2983 times.
✗ Branch 11 → 201 not taken.
✓ Branch 12 → 13 taken 106 times.
✓ Branch 12 → 29 taken 2877 times.
2983 if (conversionManager.callsOverloadedOpFct(node, 0)) {
740
1/2
✓ Branch 13 → 14 taken 106 times.
✗ Branch 13 → 166 not taken.
106 ResolverFct lhsV = [&] { return resolveValue(lhsSTy, lhs); };
741 212 ResolverFct lhsP = [&] { return resolveAddress(lhs); };
742 206 ResolverFct idxV = [&] { return resolveValue(indexExpr); };
743 112 ResolverFct idxP = [&] { return resolveAddress(indexExpr); };
744 106 lhs = conversionManager.callOperatorOverloadFct<2>(node, {lhsV, lhsP, idxV, idxP}, 0);
745 106 break;
746 106 }
747
748
1/2
✓ Branch 29 → 30 taken 2877 times.
✗ Branch 29 → 184 not taken.
2877 lhsSTy = lhsSTy.removeReferenceWrapper();
749
750 // Get the index value
751
1/2
✓ Branch 30 → 31 taken 2877 times.
✗ Branch 30 → 201 not taken.
2877 llvm::Value *indexValue = resolveValue(indexExpr);
752 // Come up with the address
753
8/10
✓ Branch 31 → 32 taken 2877 times.
✗ Branch 31 → 201 not taken.
✓ Branch 32 → 33 taken 139 times.
✓ Branch 32 → 36 taken 2738 times.
✓ Branch 33 → 34 taken 139 times.
✗ Branch 33 → 201 not taken.
✓ Branch 34 → 35 taken 98 times.
✓ Branch 34 → 36 taken 41 times.
✓ Branch 37 → 38 taken 98 times.
✓ Branch 37 → 49 taken 2779 times.
2877 if (lhsSTy.isArray() && lhsSTy.getArraySize() != ARRAY_SIZE_UNKNOWN) { // Array
754 // Make sure the address is present
755
1/2
✓ Branch 38 → 39 taken 98 times.
✗ Branch 38 → 192 not taken.
98 resolveAddress(lhs);
756
757 // Calculate address of array item
758
1/2
✓ Branch 39 → 40 taken 98 times.
✗ Branch 39 → 192 not taken.
98 llvm::Type *lhsTy = lhsSTy.toLLVMType(sourceFile);
759
1/2
✓ Branch 40 → 41 taken 98 times.
✗ Branch 40 → 192 not taken.
98 llvm::Value *indices[2] = {builder.getInt64(0), indexValue};
760
1/2
✓ Branch 45 → 46 taken 98 times.
✗ Branch 45 → 185 not taken.
98 lhs.ptr = insertInBoundsGEP(lhsTy, lhs.ptr, indices);
761 } else { // Pointer
762 // Make sure the value is present
763
1/2
✓ Branch 49 → 50 taken 2779 times.
✗ Branch 49 → 201 not taken.
2779 resolveValue(lhsNode, lhs);
764
1/2
✗ Branch 50 → 51 not taken.
✓ Branch 50 → 52 taken 2779 times.
2779 assert(lhs.value != nullptr);
765
766 // Now the pointer is the value
767 2779 lhs.ptr = lhs.value;
768
769
2/4
✓ Branch 52 → 53 taken 2779 times.
✗ Branch 52 → 193 not taken.
✓ Branch 53 → 54 taken 2779 times.
✗ Branch 53 → 193 not taken.
2779 llvm::Type *lhsTy = lhsSTy.getContained().toLLVMType(sourceFile);
770 // Calculate address of pointer item
771
1/2
✓ Branch 58 → 59 taken 2779 times.
✗ Branch 58 → 194 not taken.
2779 lhs.ptr = insertInBoundsGEP(lhsTy, lhs.ptr, indexValue);
772 }
773
774 // Reset value and entry
775 2877 lhs.value = nullptr;
776 2877 lhs.entry = nullptr;
777 2877 break;
778 }
779 14814 case PostfixUnaryExprNode::PostfixUnaryOp::OP_MEMBER_ACCESS: {
780 // Get the address of the struct instance
781
1/2
✓ Branch 64 → 65 taken 14814 times.
✗ Branch 64 → 219 not taken.
14814 resolveAddress(lhs);
782
1/2
✓ Branch 65 → 66 taken 14814 times.
✗ Branch 65 → 202 not taken.
14814 lhsSTy = lhsSTy.removeReferenceWrapper();
783
784 // Auto de-reference pointer
785
1/2
✓ Branch 66 → 67 taken 14814 times.
✗ Branch 66 → 219 not taken.
14814 autoDeReferencePtr(lhs.ptr, lhsSTy);
786
2/4
✓ Branch 67 → 68 taken 14814 times.
✗ Branch 67 → 219 not taken.
✗ Branch 68 → 69 not taken.
✓ Branch 68 → 70 taken 14814 times.
14814 assert(lhsSTy.is(TY_STRUCT));
787
788 // Retrieve struct scope
789 14814 const std::string &fieldName = node->identifier;
790
1/2
✓ Branch 70 → 71 taken 14814 times.
✗ Branch 70 → 219 not taken.
14814 Scope *structScope = lhsSTy.getBodyScope();
791
792 // Retrieve field entry
793 14814 std::vector<size_t> indexPath;
794
1/2
✓ Branch 71 → 72 taken 14814 times.
✗ Branch 71 → 217 not taken.
14814 lhs.entry = structScope->symbolTable.lookupInComposedFields(fieldName, indexPath);
795
1/2
✗ Branch 72 → 73 not taken.
✓ Branch 72 → 74 taken 14814 times.
14814 assert(lhs.entry != nullptr);
796
1/2
✓ Branch 74 → 75 taken 14814 times.
✗ Branch 74 → 217 not taken.
14814 const QualType fieldSymbolType = lhs.entry->getQualType();
797
798 // Get address of the field in the struct instance
799
2/4
✓ Branch 75 → 76 taken 14814 times.
✗ Branch 75 → 206 not taken.
✓ Branch 78 → 79 taken 14814 times.
✗ Branch 78 → 203 not taken.
29628 std::vector<llvm::Value *> indices = {builder.getInt64(0)};
800
2/2
✓ Branch 87 → 82 taken 14820 times.
✓ Branch 87 → 88 taken 14814 times.
29634 for (const size_t index : indexPath)
801
2/4
✓ Branch 83 → 84 taken 14820 times.
✗ Branch 83 → 207 not taken.
✓ Branch 84 → 85 taken 14820 times.
✗ Branch 84 → 207 not taken.
14820 indices.push_back(builder.getInt32(index));
802
1/2
✓ Branch 88 → 89 taken 14814 times.
✗ Branch 88 → 215 not taken.
14814 const std::string name = fieldName + ".addr";
803
3/6
✓ Branch 89 → 90 taken 14814 times.
✗ Branch 89 → 212 not taken.
✓ Branch 91 → 92 taken 14814 times.
✗ Branch 91 → 209 not taken.
✓ Branch 92 → 93 taken 14814 times.
✗ Branch 92 → 209 not taken.
14814 llvm::Value *memberAddr = insertInBoundsGEP(lhsSTy.toLLVMType(sourceFile), lhs.ptr, indices, name);
804
805 // Set as ptr or refPtr, depending on the type
806
3/4
✓ Branch 94 → 95 taken 14814 times.
✗ Branch 94 → 213 not taken.
✓ Branch 95 → 96 taken 189 times.
✓ Branch 95 → 97 taken 14625 times.
14814 if (fieldSymbolType.isRef()) {
807 189 lhs.ptr = nullptr;
808 189 lhs.refPtr = memberAddr;
809 } else {
810 14625 lhs.ptr = memberAddr;
811 14625 lhs.refPtr = nullptr;
812 }
813
814 // Reset the value
815 14814 lhs.value = nullptr;
816 14814 break;
817 14814 }
818 1603 case PostfixUnaryExprNode::PostfixUnaryOp::OP_PLUS_PLUS: {
819 // Make sure a value is present
820
1/2
✓ Branch 103 → 104 taken 1603 times.
✗ Branch 103 → 226 not taken.
1603 resolveValue(lhsNode, lhs);
821
822 // Allocate new local variable if required
823
2/2
✓ Branch 104 → 105 taken 2 times.
✓ Branch 104 → 115 taken 1601 times.
1603 if (!lhs.ptr) {
824
1/2
✗ Branch 105 → 106 not taken.
✓ Branch 105 → 107 taken 2 times.
2 assert(lhs.value != nullptr);
825
1/2
✓ Branch 111 → 112 taken 2 times.
✗ Branch 111 → 220 not taken.
2 lhs.ptr = insertAlloca(lhs.value->getType());
826 }
827
828 // Execute operation
829
1/2
✓ Branch 115 → 116 taken 1603 times.
✗ Branch 115 → 226 not taken.
1603 const LLVMExprResult result = conversionManager.getPostfixPlusPlusInst(node, lhs, lhsSTy, 0);
830
831 // Save the new value to the old address
832
3/4
✓ Branch 116 → 117 taken 1603 times.
✗ Branch 116 → 226 not taken.
✓ Branch 117 → 118 taken 9 times.
✓ Branch 117 → 119 taken 1594 times.
1603 if (conversionManager.callsOverloadedOpFct(node, 0)) {
833 9 lhs.value = result.value;
834 9 lhs.ptr = result.ptr;
835 } else {
836
5/6
✓ Branch 119 → 120 taken 1592 times.
✓ Branch 119 → 122 taken 2 times.
✓ Branch 120 → 121 taken 3 times.
✓ Branch 120 → 122 taken 1589 times.
✓ Branch 123 → 124 taken 1594 times.
✗ Branch 123 → 226 not taken.
1594 insertStore(result.value, lhs.ptr, lhs.entry && lhs.entry->isVolatile);
837 1594 lhs.ptr = nullptr;
838 }
839 1603 break;
840 }
841 362 case PostfixUnaryExprNode::PostfixUnaryOp::OP_MINUS_MINUS: {
842 // Make sure a value is present
843
1/2
✓ Branch 127 → 128 taken 362 times.
✗ Branch 127 → 233 not taken.
362 resolveValue(lhsNode, lhs);
844
845 // Allocate new local variable if required
846
2/2
✓ Branch 128 → 129 taken 2 times.
✓ Branch 128 → 139 taken 360 times.
362 if (!lhs.ptr) {
847
1/2
✗ Branch 129 → 130 not taken.
✓ Branch 129 → 131 taken 2 times.
2 assert(lhs.value != nullptr);
848
1/2
✓ Branch 135 → 136 taken 2 times.
✗ Branch 135 → 227 not taken.
2 lhs.ptr = insertAlloca(lhs.value->getType());
849 }
850
851 // Execute operation
852
1/2
✓ Branch 139 → 140 taken 362 times.
✗ Branch 139 → 233 not taken.
362 const LLVMExprResult result = conversionManager.getPostfixMinusMinusInst(node, lhs, lhsSTy, 0);
853
854 // Save the new value to the old address
855
3/4
✓ Branch 140 → 141 taken 362 times.
✗ Branch 140 → 233 not taken.
✓ Branch 141 → 142 taken 7 times.
✓ Branch 141 → 143 taken 355 times.
362 if (conversionManager.callsOverloadedOpFct(node, 0)) {
856 7 lhs.value = result.value;
857 7 lhs.ptr = result.ptr;
858 } else {
859
4/6
✓ Branch 143 → 144 taken 353 times.
✓ Branch 143 → 146 taken 2 times.
✗ Branch 144 → 145 not taken.
✓ Branch 144 → 146 taken 353 times.
✓ Branch 147 → 148 taken 355 times.
✗ Branch 147 → 233 not taken.
355 insertStore(result.value, lhs.ptr, lhs.entry && lhs.entry->isVolatile);
860 355 lhs.ptr = nullptr;
861 }
862 362 break;
863 }
864 default: // GCOV_EXCL_LINE
865 throw CompilerError(UNHANDLED_BRANCH, "PostfixUnaryExpr fall-through"); // GCOV_EXCL_LINE
866 }
867
868
1/2
✓ Branch 159 → 160 taken 19762 times.
✗ Branch 159 → 243 not taken.
19762 return lhs;
869
5/14
✓ Branch 17 → 18 taken 106 times.
✗ Branch 17 → 169 not taken.
✓ Branch 18 → 19 taken 106 times.
✗ Branch 18 → 169 not taken.
✓ Branch 19 → 20 taken 106 times.
✗ Branch 19 → 169 not taken.
✓ Branch 20 → 21 taken 106 times.
✗ Branch 20 → 169 not taken.
✓ Branch 21 → 22 taken 106 times.
✗ Branch 21 → 167 not taken.
✗ Branch 169 → 170 not taken.
✗ Branch 169 → 173 not taken.
✗ Branch 171 → 172 not taken.
✗ Branch 171 → 173 not taken.
106 }
870
871 79637 std::any IRGenerator::visitAtomicExpr(const AtomicExprNode *node) {
872
1/2
✓ Branch 2 → 3 taken 79637 times.
✗ Branch 2 → 89 not taken.
79637 diGenerator.setSourceLocation(node);
873
874 // If constant
875
2/2
✓ Branch 3 → 4 taken 14060 times.
✓ Branch 3 → 10 taken 65577 times.
79637 if (node->constant) {
876
2/4
✓ Branch 4 → 5 taken 14060 times.
✗ Branch 4 → 81 not taken.
✓ Branch 5 → 6 taken 14060 times.
✗ Branch 5 → 79 not taken.
14060 const auto constantValue = std::any_cast<llvm::Constant *>(visit(node->constant));
877
1/2
✓ Branch 7 → 8 taken 14060 times.
✗ Branch 7 → 82 not taken.
28120 return LLVMExprResult{.constant = constantValue};
878 }
879
880 // If value
881
2/2
✓ Branch 10 → 11 taken 16172 times.
✓ Branch 10 → 13 taken 49405 times.
65577 if (node->value)
882
1/2
✓ Branch 11 → 12 taken 16172 times.
✗ Branch 11 → 89 not taken.
16172 return visit(node->value);
883
884 // Is assign expression
885
2/2
✓ Branch 13 → 14 taken 512 times.
✓ Branch 13 → 16 taken 48893 times.
49405 if (node->assignExpr)
886
1/2
✓ Branch 14 → 15 taken 512 times.
✗ Branch 14 → 89 not taken.
512 return visit(node->assignExpr);
887
888 // Check for builtin calls
889
2/2
✓ Branch 16 → 17 taken 1580 times.
✓ Branch 16 → 19 taken 47313 times.
48893 if (node->builtinCall)
890
1/2
✓ Branch 17 → 18 taken 1580 times.
✗ Branch 17 → 89 not taken.
1580 return visit(node->builtinCall);
891
892 // Identifier (local or global variable access)
893
1/2
✗ Branch 20 → 21 not taken.
✓ Branch 20 → 22 taken 47313 times.
47313 assert(!node->identifierFragments.empty());
894
895 // Get symbol table entry
896
1/2
✓ Branch 22 → 23 taken 47313 times.
✗ Branch 22 → 89 not taken.
47313 const auto &[entry, accessScope, capture] = node->data.at(manIdx);
897
1/2
✗ Branch 23 → 24 not taken.
✓ Branch 23 → 25 taken 47313 times.
47313 assert(entry != nullptr);
898
1/2
✗ Branch 25 → 26 not taken.
✓ Branch 25 → 27 taken 47313 times.
47313 assert(accessScope != nullptr);
899
1/2
✓ Branch 27 → 28 taken 47313 times.
✗ Branch 27 → 89 not taken.
47313 const QualType varSymbolType = entry->getQualType();
900
1/2
✓ Branch 28 → 29 taken 47313 times.
✗ Branch 28 → 89 not taken.
47313 llvm::Type *varType = varSymbolType.toLLVMType(sourceFile);
901
902 // Check if external global variable
903
7/8
✓ Branch 29 → 30 taken 1430 times.
✓ Branch 29 → 33 taken 45883 times.
✓ Branch 30 → 31 taken 1430 times.
✗ Branch 30 → 89 not taken.
✓ Branch 31 → 32 taken 77 times.
✓ Branch 31 → 33 taken 1353 times.
✓ Branch 34 → 35 taken 77 times.
✓ Branch 34 → 38 taken 47236 times.
47313 if (entry->global && accessScope->isImportedBy(rootScope)) {
904 // External global variables need to be declared and allocated in the current module
905
1/2
✓ Branch 36 → 37 taken 77 times.
✗ Branch 36 → 83 not taken.
77 llvm::Value *varAddress = module->getOrInsertGlobal(entry->name, varType);
906
1/2
✓ Branch 37 → 38 taken 77 times.
✗ Branch 37 → 89 not taken.
77 entry->updateAddress(varAddress);
907 }
908
909 // Check if enum item
910
2/2
✓ Branch 38 → 39 taken 263 times.
✓ Branch 38 → 50 taken 47050 times.
47313 if (accessScope->type == ScopeType::ENUM) {
911
1/2
✓ Branch 39 → 40 taken 263 times.
✗ Branch 39 → 41 not taken.
263 const auto itemNode = spice_pointer_cast<const EnumItemNode *>(entry->declNode);
912
1/2
✓ Branch 46 → 47 taken 263 times.
✗ Branch 46 → 89 not taken.
263 llvm::Constant *constantItemValue = llvm::ConstantInt::get(varType, itemNode->itemValue);
913
1/2
✓ Branch 47 → 48 taken 263 times.
✗ Branch 47 → 84 not taken.
526 return LLVMExprResult{.constant = constantItemValue, .entry = entry};
914 }
915
916
1/2
✓ Branch 50 → 51 taken 47050 times.
✗ Branch 50 → 89 not taken.
47050 llvm::Value *address = entry->getAddress();
917
1/2
✗ Branch 51 → 52 not taken.
✓ Branch 51 → 53 taken 47050 times.
47050 assert(address != nullptr);
918
919 // If this is a function/procedure reference, return it as value
920
7/8
✓ Branch 53 → 54 taken 1430 times.
✓ Branch 53 → 57 taken 45620 times.
✓ Branch 54 → 55 taken 1430 times.
✗ Branch 54 → 85 not taken.
✓ Branch 55 → 56 taken 4 times.
✓ Branch 55 → 57 taken 1426 times.
✓ Branch 58 → 59 taken 4 times.
✓ Branch 58 → 63 taken 47046 times.
47050 if (entry->global && varSymbolType.isOneOf({TY_FUNCTION, TY_PROCEDURE})) {
921
1/2
✓ Branch 59 → 60 taken 4 times.
✗ Branch 59 → 89 not taken.
4 llvm::Value *fatPtr = buildFatFctPtr(nullptr, nullptr, address);
922
1/2
✓ Branch 60 → 61 taken 4 times.
✗ Branch 60 → 86 not taken.
8 return LLVMExprResult{.ptr = fatPtr, .entry = entry};
923 }
924
925 // Load the address of the referenced variable
926
10/12
✓ Branch 63 → 64 taken 47046 times.
✗ Branch 63 → 89 not taken.
✓ Branch 64 → 65 taken 43889 times.
✓ Branch 64 → 68 taken 3157 times.
✓ Branch 65 → 66 taken 29 times.
✓ Branch 65 → 69 taken 43860 times.
✓ Branch 66 → 67 taken 29 times.
✗ Branch 66 → 89 not taken.
✓ Branch 67 → 68 taken 11 times.
✓ Branch 67 → 69 taken 18 times.
✓ Branch 70 → 71 taken 3168 times.
✓ Branch 70 → 74 taken 43878 times.
47046 if (varSymbolType.isRef() || (capture && capture->getMode() == BY_REFERENCE))
927
1/2
✓ Branch 71 → 72 taken 3168 times.
✗ Branch 71 → 87 not taken.
6336 return LLVMExprResult{.refPtr = address, .entry = entry};
928
929
1/2
✓ Branch 74 → 75 taken 43878 times.
✗ Branch 74 → 88 not taken.
87756 return LLVMExprResult{.ptr = address, .entry = entry};
930 }
931
932 } // namespace spice::compiler
933