GCC Code Coverage Report


Directory: ../
Coverage: low: ≥ 0% medium: ≥ 75.0% high: ≥ 90.0%
Coverage Exec / Excl / Total
Lines: 96.5% 409 / 3 / 427
Functions: 92.3% 12 / 0 / 13
Branches: 53.7% 666 / 8 / 1248

src/typechecker/TypeCheckerValues.cpp
Line Branch Exec Source
1 // Copyright (c) 2021-2026 ChilliBits. All rights reserved.
2
3 #include "TypeChecker.h"
4
5 #include <SourceFile.h>
6 #include <ast/ASTNodes.h>
7 #include <ast/Attributes.h>
8 #include <global/GlobalResourceManager.h>
9 #include <model/GenericType.h>
10 #include <symboltablebuilder/Scope.h>
11 #include <symboltablebuilder/ScopeHandle.h>
12 #include <symboltablebuilder/SymbolTableBuilder.h>
13 #include <typechecker/Builtins.h>
14 #include <typechecker/FunctionManager.h>
15 #include <typechecker/MacroDefs.h>
16 #include <typechecker/TypeMatcher.h>
17
18 namespace spice::compiler {
19
20 66713 std::any TypeChecker::visitValue(ValueNode *node) {
21 // Function call
22
2/2
✓ Branch 2 → 3 taken 59739 times.
✓ Branch 2 → 4 taken 6974 times.
66713 if (node->fctCall)
23 59739 return visit(node->fctCall);
24
25 // Array initialization
26
2/2
✓ Branch 4 → 5 taken 326 times.
✓ Branch 4 → 6 taken 6648 times.
6974 if (node->arrayInitialization)
27 326 return visit(node->arrayInitialization);
28
29 // Struct instantiation
30
2/2
✓ Branch 6 → 7 taken 1312 times.
✓ Branch 6 → 8 taken 5336 times.
6648 if (node->structInstantiation)
31 1312 return visit(node->structInstantiation);
32
33 // Lambda function
34
2/2
✓ Branch 8 → 9 taken 22 times.
✓ Branch 8 → 10 taken 5314 times.
5336 if (node->lambdaFunc)
35 22 return visit(node->lambdaFunc);
36
37 // Lambda procedure
38
2/2
✓ Branch 10 → 11 taken 43 times.
✓ Branch 10 → 12 taken 5271 times.
5314 if (node->lambdaProc)
39 43 return visit(node->lambdaProc);
40
41 // Lambda expression
42
2/2
✓ Branch 12 → 13 taken 1 time.
✓ Branch 12 → 14 taken 5270 times.
5271 if (node->lambdaExpr)
43 1 return visit(node->lambdaExpr);
44
45 // Typed nil
46
1/2
✓ Branch 14 → 15 taken 5270 times.
✗ Branch 14 → 41 not taken.
5270 if (node->isNil) {
47
2/4
✓ Branch 15 → 16 taken 5270 times.
✗ Branch 15 → 52 not taken.
✓ Branch 16 → 17 taken 5270 times.
✗ Branch 16 → 50 not taken.
5270 const auto nilType = std::any_cast<QualType>(visit(node->nilType));
48
2/8
✓ Branch 18 → 19 taken 5270 times.
✗ Branch 18 → 62 not taken.
✗ Branch 19 → 20 not taken.
✓ Branch 19 → 24 taken 5270 times.
✗ Branch 20 → 21 not taken.
✗ Branch 20 → 53 not taken.
✗ Branch 21 → 22 not taken.
✗ Branch 21 → 53 not taken.
5270 HANDLE_UNRESOLVED_TYPE_ER(nilType)
49
2/4
✓ Branch 24 → 25 taken 5270 times.
✗ Branch 24 → 62 not taken.
✗ Branch 25 → 26 not taken.
✓ Branch 25 → 36 taken 5270 times.
5270 if (nilType.is(TY_DYN))
50 SOFT_ERROR_ER(node->nilType, UNEXPECTED_DYN_TYPE, "Nil must have an explicit type")
51
2/4
✓ Branch 36 → 37 taken 5270 times.
✗ Branch 36 → 61 not taken.
✓ Branch 37 → 38 taken 5270 times.
✗ Branch 37 → 61 not taken.
10540 return ExprResult{node->setEvaluatedSymbolType(nilType, manIdx)};
52 }
53
54 throw CompilerError(UNHANDLED_BRANCH, "Value fall-through"); // GCOV_EXCL_LINE
55 }
56
57 57413 std::any TypeChecker::visitConstant(ConstantNode *node) {
58 SuperType superType;
59
7/8
✓ Branch 2 → 3 taken 1813 times.
✓ Branch 2 → 4 taken 8022 times.
✓ Branch 2 → 5 taken 1130 times.
✓ Branch 2 → 6 taken 19373 times.
✓ Branch 2 → 7 taken 4737 times.
✓ Branch 2 → 8 taken 14527 times.
✓ Branch 2 → 9 taken 7811 times.
✗ Branch 2 → 10 not taken.
57413 switch (node->type) {
60 1813 case ConstantNode::PrimitiveValueType::TYPE_DOUBLE:
61 1813 superType = TY_DOUBLE;
62 1813 break;
63 8022 case ConstantNode::PrimitiveValueType::TYPE_INT:
64 8022 superType = TY_INT;
65 8022 break;
66 1130 case ConstantNode::PrimitiveValueType::TYPE_SHORT:
67 1130 superType = TY_SHORT;
68 1130 break;
69 19373 case ConstantNode::PrimitiveValueType::TYPE_LONG:
70 19373 superType = TY_LONG;
71 19373 break;
72 4737 case ConstantNode::PrimitiveValueType::TYPE_CHAR:
73 4737 superType = TY_CHAR;
74 4737 break;
75 14527 case ConstantNode::PrimitiveValueType::TYPE_STRING:
76 14527 superType = TY_STRING;
77 14527 break;
78 7811 case ConstantNode::PrimitiveValueType::TYPE_BOOL:
79 7811 superType = TY_BOOL;
80 7811 break;
81 default: // GCOV_EXCL_LINE
82 throw CompilerError(UNHANDLED_BRANCH, "Constant fall-through"); // GCOV_EXCL_LINE
83 }
84
3/6
✓ Branch 18 → 19 taken 57413 times.
✗ Branch 18 → 33 not taken.
✓ Branch 19 → 20 taken 57413 times.
✗ Branch 19 → 33 not taken.
✓ Branch 20 → 21 taken 57413 times.
✗ Branch 20 → 33 not taken.
114826 return ExprResult{node->setEvaluatedSymbolType(QualType(superType), manIdx)};
85 }
86
87 59739 std::any TypeChecker::visitFctCall(FctCallNode *node) {
88
1/2
✓ Branch 2 → 3 taken 59739 times.
✗ Branch 2 → 524 not taken.
59739 FctCallNode::FctCallData &data = node->data.at(manIdx);
89 59739 auto &[callType, isImported, templateTypes, thisType, args, callee, calleeParentScope, compTimeVal, hasCompTimeVal] = data;
90
91 // Retrieve arg types
92 59739 args.clear();
93
2/2
✓ Branch 4 → 5 taken 42041 times.
✓ Branch 4 → 36 taken 17698 times.
59739 if (node->hasArgs) {
94
1/2
✓ Branch 6 → 7 taken 42041 times.
✗ Branch 6 → 524 not taken.
42041 args.reserve(node->argLst->args.size());
95
2/2
✓ Branch 34 → 9 taken 66744 times.
✓ Branch 34 → 35 taken 42025 times.
150810 for (ExprNode *arg : node->argLst->args) {
96 // Visit argument
97
3/4
✓ Branch 11 → 12 taken 66743 times.
✓ Branch 11 → 411 taken 1 time.
✓ Branch 12 → 13 taken 66743 times.
✗ Branch 12 → 409 not taken.
66744 const auto argResult = std::any_cast<ExprResult>(visit(arg));
98
5/8
✓ Branch 14 → 15 taken 66743 times.
✗ Branch 14 → 414 not taken.
✓ Branch 15 → 16 taken 15 times.
✓ Branch 15 → 20 taken 66728 times.
✓ Branch 16 → 17 taken 15 times.
✗ Branch 16 → 412 not taken.
✓ Branch 17 → 18 taken 15 times.
✗ Branch 17 → 412 not taken.
66758 HANDLE_UNRESOLVED_TYPE_ER(argResult.type)
99
2/4
✓ Branch 20 → 21 taken 66728 times.
✗ Branch 20 → 414 not taken.
✗ Branch 21 → 22 not taken.
✓ Branch 21 → 23 taken 66728 times.
66728 assert(!argResult.type.hasAnyGenericParts());
100 // Save arg type to arg types list
101
1/2
✓ Branch 24 → 25 taken 66728 times.
✗ Branch 24 → 413 not taken.
66728 args.emplace_back(argResult.type, argResult.isTemporary());
102 }
103 }
104
105 // Retrieve template types
106 59723 templateTypes.clear();
107
2/2
✓ Branch 37 → 38 taken 4900 times.
✓ Branch 37 → 81 taken 54823 times.
59723 if (node->hasTemplateTypes) {
108
2/2
✓ Branch 79 → 40 taken 6085 times.
✓ Branch 79 → 80 taken 4900 times.
15885 for (DataTypeNode *templateTypeNode : node->templateTypeLst->dataTypes) {
109
2/4
✓ Branch 42 → 43 taken 6085 times.
✗ Branch 42 → 418 not taken.
✓ Branch 43 → 44 taken 6085 times.
✗ Branch 43 → 416 not taken.
6085 auto templateType = std::any_cast<QualType>(visit(templateTypeNode));
110
2/4
✓ Branch 45 → 46 taken 6085 times.
✗ Branch 45 → 427 not taken.
✗ Branch 46 → 47 not taken.
✓ Branch 46 → 48 taken 6085 times.
6085 assert(!templateType.is(TY_INVALID));
111
112 // Abort if the type is unresolved
113
2/4
✓ Branch 48 → 49 taken 6085 times.
✗ Branch 48 → 427 not taken.
✗ Branch 49 → 50 not taken.
✓ Branch 49 → 56 taken 6085 times.
6085 if (templateType.is(TY_UNRESOLVED))
114 HANDLE_UNRESOLVED_TYPE_ER(templateType)
115
116 // Check if the given type is generic
117
2/4
✓ Branch 56 → 57 taken 6085 times.
✗ Branch 56 → 427 not taken.
✗ Branch 57 → 58 not taken.
✓ Branch 57 → 68 taken 6085 times.
6085 if (templateType.is(TY_GENERIC))
118 SOFT_ERROR_ER(templateTypeNode, EXPECTED_NON_GENERIC_TYPE, "You must specify a concrete type here")
119
120
1/2
✓ Branch 68 → 69 taken 6085 times.
✗ Branch 68 → 427 not taken.
6085 templateTypes.push_back(templateType);
121 }
122 }
123
124 // Check if this is a builtin call
125
2/2
✓ Branch 88 → 82 taken 952862 times.
✓ Branch 88 → 89 taken 54063 times.
1006925 for (const auto &[builtinFctName, _] : BUILTIN_FUNCTIONS)
126
2/2
✓ Branch 84 → 85 taken 5660 times.
✓ Branch 84 → 87 taken 947202 times.
952862 if (node->fqFunctionName == builtinFctName)
127
1/2
✓ Branch 85 → 86 taken 5660 times.
✗ Branch 85 → 524 not taken.
5660 return visitBuiltinCall(node);
128
129 // Retrieve entry of the first fragment
130 54063 const std::string &firstFrag = node->functionNameFragments.front();
131
1/2
✓ Branch 90 → 91 taken 54063 times.
✗ Branch 90 → 524 not taken.
54063 SymbolTableEntry *firstFragEntry = currentScope->lookup(firstFrag);
132
2/2
✓ Branch 93 → 94 taken 36792 times.
✓ Branch 93 → 131 taken 17271 times.
54063 if (firstFragEntry) {
133 // Check if we have seen a 'this.' prefix, because the generator needs that
134
6/8
✓ Branch 94 → 95 taken 1 time.
✓ Branch 94 → 98 taken 36791 times.
✓ Branch 95 → 96 taken 1 time.
✗ Branch 95 → 439 not taken.
✓ Branch 96 → 97 taken 1 time.
✗ Branch 96 → 98 not taken.
✓ Branch 99 → 100 taken 1 time.
✓ Branch 99 → 109 taken 36791 times.
36792 if (firstFragEntry->scope->type == ScopeType::STRUCT && firstFrag != THIS_VARIABLE_NAME)
135
5/10
✓ Branch 100 → 101 taken 1 time.
✗ Branch 100 → 433 not taken.
✓ Branch 101 → 102 taken 1 time.
✗ Branch 101 → 431 not taken.
✓ Branch 102 → 103 taken 1 time.
✗ Branch 102 → 429 not taken.
✓ Branch 105 → 106 taken 1 time.
✗ Branch 105 → 435 not taken.
✓ Branch 106 → 107 taken 1 time.
✗ Branch 106 → 435 not taken.
2 SOFT_ERROR_ER(node, REFERENCED_UNDEFINED_VARIABLE,
136 "The symbol '" + firstFrag + "' could not be found. Missing 'this.' prefix?")
137
138 36791 firstFragEntry->used = true;
139 // Decide of which type the function call is
140
2/4
✓ Branch 109 → 110 taken 36791 times.
✗ Branch 109 → 439 not taken.
✓ Branch 110 → 111 taken 36791 times.
✗ Branch 110 → 439 not taken.
36791 const QualType &baseType = firstFragEntry->getQualType().getBase();
141
5/8
✓ Branch 111 → 112 taken 36791 times.
✗ Branch 111 → 439 not taken.
✓ Branch 112 → 113 taken 4 times.
✓ Branch 112 → 117 taken 36787 times.
✓ Branch 113 → 114 taken 4 times.
✗ Branch 113 → 436 not taken.
✓ Branch 114 → 115 taken 4 times.
✗ Branch 114 → 436 not taken.
36795 HANDLE_UNRESOLVED_TYPE_ER(baseType)
142
3/4
✓ Branch 117 → 118 taken 36787 times.
✗ Branch 117 → 437 not taken.
✓ Branch 118 → 119 taken 30206 times.
✓ Branch 118 → 122 taken 6581 times.
36787 if (baseType.isOneOf({TY_STRUCT, TY_INTERFACE})) {
143
2/2
✓ Branch 119 → 120 taken 3855 times.
✓ Branch 119 → 121 taken 26351 times.
30206 if (firstFragEntry->scope->type == ScopeType::GLOBAL)
144 3855 callType = FctCallNode::FctCallType::TYPE_CTOR;
145 else
146 26351 callType = FctCallNode::FctCallType::TYPE_METHOD;
147
7/8
✓ Branch 122 → 123 taken 6581 times.
✗ Branch 122 → 438 not taken.
✓ Branch 123 → 124 taken 6423 times.
✓ Branch 123 → 126 taken 158 times.
✓ Branch 124 → 125 taken 102 times.
✓ Branch 124 → 126 taken 6321 times.
✓ Branch 127 → 128 taken 102 times.
✓ Branch 127 → 129 taken 6479 times.
6581 } else if (baseType.isOneOf({TY_FUNCTION, TY_PROCEDURE}) && firstFragEntry->scope->type != ScopeType::GLOBAL) {
148 102 callType = FctCallNode::FctCallType::TYPE_FCT_PTR;
149 }
150 }
151
152 // Get struct name. Retrieve it from alias if required
153
1/2
✓ Branch 131 → 132 taken 54058 times.
✗ Branch 131 → 524 not taken.
54058 const auto &[structEntry, isAlias] = rootScope->symbolTable.lookupWithAliasResolution(node->fqFunctionName);
154
4/6
✓ Branch 134 → 135 taken 1 time.
✓ Branch 134 → 138 taken 54057 times.
✓ Branch 135 → 136 taken 1 time.
✗ Branch 135 → 524 not taken.
✓ Branch 136 → 137 taken 1 time.
✗ Branch 136 → 524 not taken.
54058 const std::string &fqFunctionName = isAlias ? structEntry->getQualType().getSubType() : node->fqFunctionName;
155
156 // Get the concrete template types
157
2/2
✓ Branch 139 → 140 taken 1 time.
✓ Branch 139 → 159 taken 54057 times.
54058 if (isAlias) {
158 // Retrieve concrete template types from type alias
159
3/6
✓ Branch 140 → 141 taken 1 time.
✗ Branch 140 → 524 not taken.
✓ Branch 141 → 142 taken 1 time.
✗ Branch 141 → 524 not taken.
✓ Branch 142 → 143 taken 1 time.
✗ Branch 142 → 524 not taken.
1 templateTypes = structEntry->getQualType().getTemplateTypes();
160 // Check if the aliased type specified template types and the struct instantiation does
161
3/6
✓ Branch 144 → 145 taken 1 time.
✗ Branch 144 → 147 not taken.
✗ Branch 145 → 146 not taken.
✓ Branch 145 → 147 taken 1 time.
✗ Branch 148 → 149 not taken.
✓ Branch 148 → 159 taken 1 time.
1 if (!templateTypes.empty() && node->hasTemplateTypes)
162 SOFT_ERROR_ER(node->templateTypeLst, ALIAS_WITH_TEMPLATE_LIST, "The aliased type already has a template list")
163 }
164
165 // Check if this is a method call or a normal function call
166
2/2
✓ Branch 160 → 161 taken 26351 times.
✓ Branch 160 → 175 taken 27707 times.
54058 if (data.isMethodCall()) {
167 // This is a method call
168
1/2
✓ Branch 161 → 162 taken 26351 times.
✗ Branch 161 → 524 not taken.
26351 thisType = firstFragEntry->getQualType();
169
2/4
✓ Branch 162 → 163 taken 26351 times.
✗ Branch 162 → 447 not taken.
✓ Branch 163 → 164 taken 26351 times.
✗ Branch 163 → 447 not taken.
26351 Scope *structBodyScope = thisType.getBase().getBodyScope();
170
1/2
✗ Branch 164 → 165 not taken.
✓ Branch 164 → 166 taken 26351 times.
26351 assert(structBodyScope != nullptr);
171
3/4
✓ Branch 166 → 167 taken 26351 times.
✗ Branch 166 → 524 not taken.
✓ Branch 167 → 168 taken 2 times.
✓ Branch 167 → 173 taken 26349 times.
26351 if (!visitMethodCall(node, structBodyScope)) // Check if soft errors occurred
172
3/6
✓ Branch 168 → 169 taken 2 times.
✗ Branch 168 → 448 not taken.
✓ Branch 169 → 170 taken 2 times.
✗ Branch 169 → 448 not taken.
✓ Branch 170 → 171 taken 2 times.
✗ Branch 170 → 448 not taken.
4 return ExprResult{node->setEvaluatedSymbolType(QualType(TY_UNRESOLVED), manIdx)};
173
1/2
✗ Branch 173 → 174 not taken.
✓ Branch 173 → 211 taken 26349 times.
26349 assert(calleeParentScope != nullptr);
174
2/2
✓ Branch 176 → 177 taken 102 times.
✓ Branch 176 → 190 taken 27605 times.
27707 } else if (data.isFctPtrCall()) {
175 // This is a function pointer call
176
2/4
✓ Branch 177 → 178 taken 102 times.
✗ Branch 177 → 453 not taken.
✓ Branch 178 → 179 taken 102 times.
✗ Branch 178 → 453 not taken.
102 const QualType &functionType = firstFragEntry->getQualType().getBase();
177
2/4
✓ Branch 179 → 180 taken 102 times.
✗ Branch 179 → 450 not taken.
✗ Branch 180 → 181 not taken.
✓ Branch 180 → 182 taken 102 times.
102 assert(functionType.isOneOf({TY_FUNCTION, TY_PROCEDURE}));
178
2/4
✓ Branch 182 → 183 taken 102 times.
✗ Branch 182 → 453 not taken.
✗ Branch 183 → 184 not taken.
✓ Branch 183 → 189 taken 102 times.
102 if (!visitFctPtrCall(node, functionType)) // Check if soft errors occurred
179 return ExprResult{node->setEvaluatedSymbolType(QualType(TY_UNRESOLVED), manIdx)};
180 } else {
181 // This is an ordinary function call
182
3/4
✓ Branch 191 → 192 taken 3890 times.
✓ Branch 191 → 195 taken 23715 times.
✗ Branch 193 → 194 not taken.
✓ Branch 193 → 195 taken 3890 times.
27605 assert(data.isOrdinaryCall() || data.isCtorCall());
183
5/6
✓ Branch 195 → 196 taken 27605 times.
✗ Branch 195 → 456 not taken.
✓ Branch 196 → 197 taken 27604 times.
✓ Branch 196 → 454 taken 1 time.
✓ Branch 198 → 199 taken 4 times.
✓ Branch 198 → 204 taken 27600 times.
27606 if (!visitOrdinaryFctCall(node, fqFunctionName)) // Check if soft errors occurred
184
3/6
✓ Branch 199 → 200 taken 4 times.
✗ Branch 199 → 457 not taken.
✓ Branch 200 → 201 taken 4 times.
✗ Branch 200 → 457 not taken.
✓ Branch 201 → 202 taken 4 times.
✗ Branch 201 → 457 not taken.
8 return ExprResult{node->setEvaluatedSymbolType(QualType(TY_UNRESOLVED), manIdx)};
185
1/2
✗ Branch 204 → 205 not taken.
✓ Branch 204 → 206 taken 27600 times.
27600 assert(calleeParentScope != nullptr);
186
187 // If the call is no ordinary call, it must be a constructor, which takes a struct as this type.
188
4/6
✓ Branch 207 → 208 taken 9926 times.
✓ Branch 207 → 211 taken 17674 times.
✓ Branch 208 → 209 taken 9926 times.
✗ Branch 208 → 524 not taken.
✗ Branch 209 → 210 not taken.
✓ Branch 209 → 211 taken 9926 times.
27600 assert(data.isOrdinaryCall() || data.thisType.is(TY_STRUCT));
189 }
190
191
2/2
✓ Branch 212 → 213 taken 53949 times.
✓ Branch 212 → 284 taken 102 times.
54051 if (!data.isFctPtrCall()) {
192 // Check if we were able to find a function
193
2/2
✓ Branch 213 → 214 taken 12 times.
✓ Branch 213 → 251 taken 53937 times.
53949 if (!callee) {
194 // Build error message
195
6/10
✓ Branch 215 → 216 taken 2 times.
✓ Branch 215 → 219 taken 10 times.
✓ Branch 218 → 221 taken 2 times.
✗ Branch 218 → 459 not taken.
✓ Branch 220 → 221 taken 10 times.
✗ Branch 220 → 459 not taken.
✓ Branch 221 → 222 taken 2 times.
✓ Branch 221 → 224 taken 10 times.
✗ Branch 459 → 460 not taken.
✗ Branch 459 → 462 not taken.
14 const std::string functionName = data.isCtorCall() ? CTOR_FUNCTION_NAME : node->functionNameFragments.back();
196 12 ParamList errArgTypes;
197
1/2
✓ Branch 225 → 226 taken 12 times.
✗ Branch 225 → 479 not taken.
12 errArgTypes.reserve(args.size());
198
5/8
✓ Branch 226 → 227 taken 12 times.
✗ Branch 226 → 465 not taken.
✓ Branch 227 → 228 taken 12 times.
✗ Branch 227 → 465 not taken.
✓ Branch 228 → 229 taken 12 times.
✗ Branch 228 → 465 not taken.
✓ Branch 234 → 230 taken 7 times.
✓ Branch 234 → 235 taken 12 times.
19 for (const auto &type : args | std::views::keys)
199
1/2
✓ Branch 231 → 232 taken 7 times.
✗ Branch 231 → 464 not taken.
7 errArgTypes.push_back({type, false});
200
2/4
✓ Branch 236 → 237 taken 12 times.
✗ Branch 236 → 466 not taken.
✓ Branch 237 → 238 taken 12 times.
✗ Branch 237 → 466 not taken.
12 const std::string signature = Function::getSignature(functionName, thisType, QualType(TY_DYN), errArgTypes, {}, false);
201 // Throw error
202
5/10
✓ Branch 239 → 240 taken 12 times.
✗ Branch 239 → 474 not taken.
✓ Branch 240 → 241 taken 12 times.
✗ Branch 240 → 472 not taken.
✓ Branch 241 → 242 taken 12 times.
✗ Branch 241 → 470 not taken.
✓ Branch 244 → 245 taken 12 times.
✗ Branch 244 → 476 not taken.
✓ Branch 245 → 246 taken 12 times.
✗ Branch 245 → 476 not taken.
12 SOFT_ERROR_ER(node, REFERENCED_UNDEFINED_FUNCTION, "Function/procedure '" + signature + "' could not be found")
203 12 }
204
205 // Check if we need to request a re-visit, because the function body was not type-checked yet
206
1/2
✓ Branch 251 → 252 taken 53937 times.
✗ Branch 251 → 524 not taken.
53937 requestRevisitIfRequired(callee);
207
208 // Get function entry from function object
209 53937 SymbolTableEntry *functionEntry = callee->entry;
210
211 // Check if the called function has sufficient visibility
212
1/2
✓ Branch 252 → 253 taken 53937 times.
✗ Branch 252 → 524 not taken.
53937 isImported = calleeParentScope->isImportedBy(rootScope);
213
8/10
✓ Branch 253 → 254 taken 22085 times.
✓ Branch 253 → 258 taken 31852 times.
✓ Branch 254 → 255 taken 22085 times.
✗ Branch 254 → 524 not taken.
✓ Branch 255 → 256 taken 22085 times.
✗ Branch 255 → 524 not taken.
✓ Branch 256 → 257 taken 1 time.
✓ Branch 256 → 258 taken 22084 times.
✓ Branch 259 → 260 taken 1 time.
✓ Branch 259 → 284 taken 53936 times.
53937 if (isImported && !functionEntry->getQualType().isPublic()) {
214
1/2
✓ Branch 260 → 261 taken 1 time.
✗ Branch 260 → 500 not taken.
1 const QualType functionEntryType = functionEntry->getQualType();
215
1/2
✓ Branch 261 → 262 taken 1 time.
✗ Branch 261 → 500 not taken.
1 const std::string signature = callee->getSignature();
216
2/4
✓ Branch 262 → 263 taken 1 time.
✗ Branch 262 → 498 not taken.
✓ Branch 263 → 264 taken 1 time.
✗ Branch 263 → 273 not taken.
1 if (functionEntryType.is(TY_FUNCTION))
217
5/10
✓ Branch 264 → 265 taken 1 time.
✗ Branch 264 → 488 not taken.
✓ Branch 265 → 266 taken 1 time.
✗ Branch 265 → 486 not taken.
✓ Branch 266 → 267 taken 1 time.
✗ Branch 266 → 484 not taken.
✓ Branch 269 → 270 taken 1 time.
✗ Branch 269 → 490 not taken.
✓ Branch 270 → 271 taken 1 time.
✗ Branch 270 → 490 not taken.
2 SOFT_ERROR_ER(node, INSUFFICIENT_VISIBILITY, "Function '" + signature + "' has insufficient visibility")
218 else
219 SOFT_ERROR_ER(node, INSUFFICIENT_VISIBILITY, "Procedure '" + signature + "' has insufficient visibility")
220 1 }
221 }
222
223 // Generate arg infos
224
2/2
✓ Branch 284 → 285 taken 37639 times.
✓ Branch 284 → 327 taken 16399 times.
54038 if (node->hasArgs) {
225 37639 QualTypeList paramTypes;
226
2/2
✓ Branch 286 → 287 taken 76 times.
✓ Branch 286 → 293 taken 37563 times.
37639 if (data.isFctPtrCall()) {
227
2/4
✓ Branch 287 → 288 taken 76 times.
✗ Branch 287 → 502 not taken.
✓ Branch 288 → 289 taken 76 times.
✗ Branch 288 → 502 not taken.
76 const QualType &functionType = firstFragEntry->getQualType().getBase();
228
1/2
✓ Branch 289 → 290 taken 76 times.
✗ Branch 289 → 501 not taken.
76 paramTypes = functionType.getFunctionParamTypes();
229 } else {
230
1/2
✗ Branch 293 → 294 not taken.
✓ Branch 293 → 295 taken 37563 times.
37563 assert(callee != nullptr);
231
1/2
✓ Branch 295 → 296 taken 37563 times.
✗ Branch 295 → 503 not taken.
37563 paramTypes = callee->getParamTypes();
232 }
233
234 37639 node->argLst->argInfos.clear();
235
2/2
✓ Branch 320 → 301 taken 60858 times.
✓ Branch 320 → 321 taken 37639 times.
98497 for (size_t argIdx = 0; argIdx < args.size(); argIdx++) {
236
1/2
✓ Branch 301 → 302 taken 60858 times.
✗ Branch 301 → 505 not taken.
60858 const QualType &expectedType = paramTypes.at(argIdx);
237
1/2
✓ Branch 302 → 303 taken 60858 times.
✗ Branch 302 → 505 not taken.
60858 const auto &[actualType, _] = args.at(argIdx);
238
239 60858 Function *copyCtor = nullptr;
240
11/14
✓ Branch 305 → 306 taken 60858 times.
✗ Branch 305 → 505 not taken.
✓ Branch 306 → 307 taken 1114 times.
✓ Branch 306 → 312 taken 59744 times.
✓ Branch 307 → 308 taken 1114 times.
✗ Branch 307 → 505 not taken.
✓ Branch 308 → 309 taken 1110 times.
✓ Branch 308 → 312 taken 4 times.
✓ Branch 309 → 310 taken 1110 times.
✗ Branch 309 → 505 not taken.
✓ Branch 310 → 311 taken 47 times.
✓ Branch 310 → 312 taken 1063 times.
✓ Branch 313 → 314 taken 47 times.
✓ Branch 313 → 317 taken 60811 times.
60858 if (expectedType.is(TY_STRUCT) && actualType.is(TY_STRUCT) && !actualType.isTriviallyCopyable(node)) {
241
1/2
✓ Branch 314 → 315 taken 47 times.
✗ Branch 314 → 505 not taken.
47 copyCtor = matchCopyCtor(actualType, node);
242 // Insert anonymous symbol to track the dtor call of the copy
243
1/2
✓ Branch 315 → 316 taken 47 times.
✗ Branch 315 → 505 not taken.
47 ExprNode *argNode = node->argLst->args.at(argIdx);
244
1/2
✓ Branch 316 → 317 taken 47 times.
✗ Branch 316 → 505 not taken.
47 currentScope->symbolTable.insertAnonymous(actualType, argNode, SIZE_MAX);
245 }
246
247
1/2
✓ Branch 317 → 318 taken 60858 times.
✗ Branch 317 → 504 not taken.
60858 node->argLst->argInfos.push_back(ArgLstNode::ArgInfo{copyCtor});
248 }
249
1/2
✗ Branch 323 → 324 not taken.
✓ Branch 323 → 325 taken 37639 times.
37639 assert(node->argLst->argInfos.size() == node->argLst->args.size());
250 37639 }
251
252 // Retrieve return type
253
6/10
✓ Branch 328 → 329 taken 102 times.
✓ Branch 328 → 333 taken 53936 times.
✓ Branch 329 → 330 taken 102 times.
✗ Branch 329 → 508 not taken.
✓ Branch 330 → 331 taken 102 times.
✗ Branch 330 → 508 not taken.
✓ Branch 331 → 332 taken 102 times.
✗ Branch 331 → 508 not taken.
✓ Branch 333 → 334 taken 53936 times.
✗ Branch 333 → 508 not taken.
107974 const bool isFct = data.isFctPtrCall() ? firstFragEntry->getQualType().getBase().is(TY_FUNCTION) : callee->isFunction();
254 54038 QualType returnType;
255
2/2
✓ Branch 338 → 339 taken 102 times.
✓ Branch 338 → 346 taken 53936 times.
54038 if (data.isFctPtrCall()) {
256
6/10
✓ Branch 339 → 340 taken 25 times.
✓ Branch 339 → 344 taken 77 times.
✓ Branch 340 → 341 taken 25 times.
✗ Branch 340 → 509 not taken.
✓ Branch 341 → 342 taken 25 times.
✗ Branch 341 → 509 not taken.
✓ Branch 342 → 343 taken 25 times.
✗ Branch 342 → 509 not taken.
✓ Branch 344 → 345 taken 77 times.
✗ Branch 344 → 509 not taken.
102 returnType = isFct ? firstFragEntry->getQualType().getBase().getFunctionReturnType() : QualType(TY_BOOL);
257
2/2
✓ Branch 347 → 348 taken 9924 times.
✓ Branch 347 → 349 taken 44012 times.
53936 } else if (data.isCtorCall()) {
258 9924 returnType = thisType;
259
3/4
✓ Branch 349 → 350 taken 44012 times.
✗ Branch 349 → 524 not taken.
✓ Branch 352 → 353 taken 11254 times.
✓ Branch 352 → 355 taken 32758 times.
88024 } else if (callee->isProcedure()) {
260
1/2
✓ Branch 353 → 354 taken 11254 times.
✗ Branch 353 → 511 not taken.
11254 returnType = QualType(TY_DYN);
261 } else {
262 32758 returnType = callee->returnType;
263 }
264
265
1/2
✓ Branch 356 → 357 taken 54038 times.
✗ Branch 356 → 524 not taken.
54038 const QualType returnBaseType = returnType.getBase();
266
267 // Make sure this source file knows about the return type
268
3/4
✓ Branch 357 → 358 taken 54038 times.
✗ Branch 357 → 524 not taken.
✓ Branch 358 → 359 taken 17997 times.
✓ Branch 358 → 362 taken 36041 times.
54038 if (returnBaseType.is(TY_STRUCT))
269
2/4
✓ Branch 359 → 360 taken 17997 times.
✗ Branch 359 → 512 not taken.
✓ Branch 360 → 361 taken 17997 times.
✗ Branch 360 → 512 not taken.
17997 returnType = mapImportedScopeTypeToLocalType(returnBaseType.getBodyScope(), returnType);
270
271 // Add anonymous symbol to keep track of dtor call, if non-trivially destructible
272 54038 SymbolTableEntry *anonymousSymbol = nullptr;
273
8/10
✓ Branch 362 → 363 taken 54038 times.
✗ Branch 362 → 524 not taken.
✓ Branch 363 → 364 taken 13840 times.
✓ Branch 363 → 367 taken 40198 times.
✓ Branch 364 → 365 taken 13840 times.
✗ Branch 364 → 524 not taken.
✓ Branch 365 → 366 taken 9246 times.
✓ Branch 365 → 367 taken 4594 times.
✓ Branch 368 → 369 taken 9246 times.
✓ Branch 368 → 371 taken 44792 times.
54038 if (returnType.is(TY_STRUCT) && !returnType.isTriviallyDestructible(node))
274
1/2
✓ Branch 369 → 370 taken 9246 times.
✗ Branch 369 → 524 not taken.
9246 anonymousSymbol = currentScope->symbolTable.insertAnonymous(returnType, node);
275
276 // Remove public qualifier to not have public local variables
277 54038 returnType.getQualifiers().isPublic = false;
278
279 // Check if the return value gets discarded
280
7/8
✓ Branch 372 → 373 taken 32783 times.
✓ Branch 372 → 376 taken 21255 times.
✓ Branch 373 → 374 taken 32783 times.
✗ Branch 373 → 524 not taken.
✓ Branch 374 → 375 taken 1567 times.
✓ Branch 374 → 376 taken 31216 times.
✓ Branch 377 → 378 taken 1567 times.
✓ Branch 377 → 403 taken 52471 times.
54038 if (isFct && !node->hasReturnValueReceiver()) {
281 // Check if we want to ignore the discarded return value
282 1567 bool ignoreUnusedReturnValue = false;
283
2/2
✓ Branch 379 → 380 taken 1564 times.
✓ Branch 379 → 400 taken 3 times.
1567 if (!data.isFctPtrCall()) {
284
1/2
✗ Branch 380 → 381 not taken.
✓ Branch 380 → 382 taken 1564 times.
1564 assert(callee != nullptr);
285
1/2
✓ Branch 382 → 383 taken 1564 times.
✗ Branch 382 → 384 not taken.
1564 auto fctDef = dynamic_cast<const FctDefNode *>(callee->declNode);
286
12/18
✓ Branch 385 → 386 taken 1259 times.
✓ Branch 385 → 393 taken 305 times.
✓ Branch 386 → 387 taken 283 times.
✓ Branch 386 → 393 taken 976 times.
✓ Branch 389 → 390 taken 283 times.
✗ Branch 389 → 513 not taken.
✓ Branch 390 → 391 taken 283 times.
✗ Branch 390 → 513 not taken.
✓ Branch 391 → 392 taken 264 times.
✓ Branch 391 → 393 taken 19 times.
✓ Branch 394 → 395 taken 283 times.
✓ Branch 394 → 396 taken 1281 times.
✓ Branch 396 → 397 taken 283 times.
✓ Branch 396 → 399 taken 1281 times.
✗ Branch 513 → 514 not taken.
✗ Branch 513 → 515 not taken.
✗ Branch 517 → 518 not taken.
✗ Branch 517 → 520 not taken.
2130 ignoreUnusedReturnValue = fctDef && fctDef->attrs && fctDef->attrs->attrLst->hasAttr(ATTR_IGNORE_UNUSED_RETURN_VALUE);
287 }
288
289
2/2
✓ Branch 400 → 401 taken 1303 times.
✓ Branch 400 → 403 taken 264 times.
1567 if (!ignoreUnusedReturnValue)
290
1/2
✓ Branch 401 → 402 taken 1303 times.
✗ Branch 401 → 522 not taken.
1303 warnings.emplace_back(node->codeLoc, UNUSED_RETURN_VALUE, "The return value of the function call is unused");
291 }
292
293
2/4
✓ Branch 403 → 404 taken 54038 times.
✗ Branch 403 → 523 not taken.
✓ Branch 404 → 405 taken 54038 times.
✗ Branch 404 → 523 not taken.
108076 return ExprResult{node->setEvaluatedSymbolType(returnType, manIdx), anonymousSymbol};
294 }
295
296 27605 bool TypeChecker::visitOrdinaryFctCall(FctCallNode *node, std::string fqFunctionName) const {
297
1/2
✓ Branch 2 → 3 taken 27605 times.
✗ Branch 2 → 112 not taken.
27605 FctCallNode::FctCallData &data = node->data.at(manIdx);
298 27605 auto &[callType, isImported, templateTypes, thisType, args, callee, calleeParentScope, compTimeVal, hasCompTimeVal] = data;
299
300 // Check if this is a well-known ctor/fct call
301
2/2
✓ Branch 4 → 5 taken 27453 times.
✓ Branch 4 → 7 taken 152 times.
27605 if (node->functionNameFragments.size() == 1) {
302
1/2
✓ Branch 5 → 6 taken 27453 times.
✗ Branch 5 → 112 not taken.
27453 ensureLoadedRuntimeForTypeName(fqFunctionName);
303
1/2
✓ Branch 6 → 7 taken 27453 times.
✗ Branch 6 → 112 not taken.
27453 ensureLoadedRuntimeForFunctionName(fqFunctionName);
304 }
305
306 // Check if the type is generic (possible in case of ctor call)
307
1/2
✓ Branch 7 → 8 taken 27605 times.
✗ Branch 7 → 112 not taken.
27605 const QualType *genericType = rootScope->lookupGenericTypeStrict(fqFunctionName);
308
6/8
✓ Branch 8 → 9 taken 1 time.
✓ Branch 8 → 12 taken 27604 times.
✓ Branch 9 → 10 taken 1 time.
✗ Branch 9 → 112 not taken.
✓ Branch 10 → 11 taken 1 time.
✗ Branch 10 → 12 not taken.
✓ Branch 13 → 14 taken 1 time.
✓ Branch 13 → 19 taken 27604 times.
27605 if (genericType && typeMapping.contains(fqFunctionName)) {
309
1/2
✓ Branch 14 → 15 taken 1 time.
✗ Branch 14 → 112 not taken.
1 const QualType &replacementType = typeMapping.at(fqFunctionName);
310
2/4
✓ Branch 15 → 16 taken 1 time.
✗ Branch 15 → 112 not taken.
✓ Branch 16 → 17 taken 1 time.
✗ Branch 16 → 19 not taken.
1 if (replacementType.is(TY_STRUCT))
311
2/4
✓ Branch 17 → 18 taken 1 time.
✗ Branch 17 → 112 not taken.
✓ Branch 18 → 19 taken 1 time.
✗ Branch 18 → 112 not taken.
1 fqFunctionName = replacementType.getSubType();
312 }
313
314 // Check if the exported name registry contains that function name
315
1/2
✓ Branch 19 → 20 taken 27605 times.
✗ Branch 19 → 112 not taken.
27605 const NameRegistryEntry *functionRegistryEntry = sourceFile->getNameRegistryEntry(fqFunctionName);
316
2/2
✓ Branch 20 → 21 taken 3 times.
✓ Branch 20 → 28 taken 27602 times.
27605 if (!functionRegistryEntry) {
317
2/4
✓ Branch 22 → 23 taken 3 times.
✗ Branch 22 → 92 not taken.
✓ Branch 23 → 24 taken 3 times.
✗ Branch 23 → 90 not taken.
3 const std::string msg = "Function/procedure/struct '" + node->functionNameFragments.back() + "' could not be found";
318
1/2
✓ Branch 25 → 26 taken 3 times.
✗ Branch 25 → 93 not taken.
3 SOFT_ERROR_BOOL(node, REFERENCED_UNDEFINED_FUNCTION, msg)
319 3 }
320 27602 const SymbolTableEntry *functionEntry = functionRegistryEntry->targetEntry;
321 27602 calleeParentScope = functionRegistryEntry->targetScope;
322
323 // Check if the target symbol is a struct -> this must be a constructor call
324
1/2
✓ Branch 29 → 30 taken 27602 times.
✗ Branch 29 → 112 not taken.
27602 std::string functionName = node->functionNameFragments.back();
325
7/10
✓ Branch 30 → 31 taken 27602 times.
✗ Branch 30 → 35 not taken.
✓ Branch 31 → 32 taken 27602 times.
✗ Branch 31 → 110 not taken.
✓ Branch 32 → 33 taken 27602 times.
✗ Branch 32 → 110 not taken.
✓ Branch 33 → 34 taken 9927 times.
✓ Branch 33 → 35 taken 17675 times.
✓ Branch 36 → 37 taken 9927 times.
✓ Branch 36 → 52 taken 17675 times.
27602 if (functionEntry != nullptr && functionEntry->getQualType().is(TY_STRUCT)) {
326 9927 callType = FctCallNode::FctCallType::TYPE_CTOR;
327
1/2
✓ Branch 37 → 38 taken 9927 times.
✗ Branch 37 → 110 not taken.
9927 functionName = CTOR_FUNCTION_NAME;
328
329 9927 const NameRegistryEntry *structRegistryEntry = functionRegistryEntry;
330 9927 const SymbolTableEntry *structEntry = functionEntry;
331
332 // Substantiate potentially generic this struct
333
2/4
✓ Branch 38 → 39 taken 9927 times.
✗ Branch 38 → 110 not taken.
✓ Branch 39 → 40 taken 9927 times.
✗ Branch 39 → 110 not taken.
9927 const Struct *thisStruct = structEntry->getQualType().getStruct(node, templateTypes);
334
2/2
✓ Branch 40 → 41 taken 1 time.
✓ Branch 40 → 49 taken 9926 times.
9927 if (!thisStruct) {
335
1/2
✓ Branch 41 → 42 taken 1 time.
✗ Branch 41 → 103 not taken.
1 const std::string signature = Struct::getSignature(structRegistryEntry->targetEntry->name, templateTypes);
336
2/4
✓ Branch 42 → 43 taken 1 time.
✗ Branch 42 → 98 not taken.
✓ Branch 43 → 44 taken 1 time.
✗ Branch 43 → 96 not taken.
1 const std::string errorMsg = "Could not find struct candidate for struct '" + signature + "'. Do the template types match?";
337
1/2
✓ Branch 45 → 46 taken 1 time.
✗ Branch 45 → 99 not taken.
1 SOFT_ERROR_BOOL(node, UNKNOWN_DATATYPE, errorMsg)
338 1 }
339
340 // Set the 'this' type of the function to the struct type
341
2/4
✓ Branch 49 → 50 taken 9926 times.
✗ Branch 49 → 104 not taken.
✓ Branch 50 → 51 taken 9926 times.
✗ Branch 50 → 104 not taken.
9926 thisType = structEntry->getQualType().getWithBodyScope(thisStruct->scope);
342 9926 calleeParentScope = thisStruct->scope;
343 }
344
345 // Attach the concrete template types to the 'this' type
346
7/8
✓ Branch 52 → 53 taken 27601 times.
✗ Branch 52 → 110 not taken.
✓ Branch 53 → 54 taken 9926 times.
✓ Branch 53 → 57 taken 17675 times.
✓ Branch 55 → 56 taken 1755 times.
✓ Branch 55 → 57 taken 8171 times.
✓ Branch 58 → 59 taken 1755 times.
✓ Branch 58 → 61 taken 25846 times.
27601 if (!thisType.is(TY_DYN) && !templateTypes.empty())
347
1/2
✓ Branch 59 → 60 taken 1755 times.
✗ Branch 59 → 105 not taken.
1755 thisType = thisType.getWithTemplateTypes(templateTypes);
348
349 // Map local arg types to imported types
350
5/8
✓ Branch 61 → 62 taken 27601 times.
✗ Branch 61 → 107 not taken.
✓ Branch 62 → 63 taken 27601 times.
✗ Branch 62 → 107 not taken.
✓ Branch 63 → 64 taken 27601 times.
✗ Branch 63 → 107 not taken.
✓ Branch 69 → 65 taken 44698 times.
✓ Branch 69 → 70 taken 27601 times.
72299 for (QualType &argType : args | std::views::keys)
351
1/2
✓ Branch 66 → 67 taken 44698 times.
✗ Branch 66 → 106 not taken.
44698 argType = mapLocalTypeToImportedScopeType(calleeParentScope, argType);
352
353 // Map local template types to imported types
354
2/2
✓ Branch 84 → 72 taken 3891 times.
✓ Branch 84 → 85 taken 27601 times.
59093 for (QualType &templateType : templateTypes)
355
1/2
✓ Branch 74 → 75 taken 3891 times.
✗ Branch 74 → 108 not taken.
3891 templateType = mapLocalTypeToImportedScopeType(calleeParentScope, templateType);
356
357 // Retrieve function object
358 27601 Scope *matchScope = calleeParentScope;
359
2/2
✓ Branch 85 → 86 taken 27600 times.
✓ Branch 85 → 110 taken 1 time.
27601 callee = FunctionManager::match(matchScope, functionName, data.thisType, data.args, templateTypes, false, node);
360
361 27600 return true;
362 27602 }
363
364 102 bool TypeChecker::visitFctPtrCall(const FctCallNode *node, const QualType &functionType) const {
365
1/2
✓ Branch 2 → 3 taken 102 times.
✗ Branch 2 → 75 not taken.
102 const FctCallNode::FctCallData &data = node->data.at(manIdx);
366 102 const auto &[callType, isImported, templateTypes, thisType, args, callee, calleeParentScope, compTimeVal, hasCompTimeVal] =
367 data;
368
369 // Check if the given argument types match the type
370
1/2
✓ Branch 3 → 4 taken 102 times.
✗ Branch 3 → 75 not taken.
102 const QualTypeList expectedArgTypes = functionType.getFunctionParamTypes();
371
1/2
✗ Branch 6 → 7 not taken.
✓ Branch 6 → 14 taken 102 times.
102 if (args.size() != expectedArgTypes.size())
372 SOFT_ERROR_BOOL(node, REFERENCED_UNDEFINED_FUNCTION, "Expected and actual number of arguments do not match")
373
374 // Create resolver function, that always returns a nullptr
375 102 TypeMatcher::ResolverFct resolverFct = [](const std::string &) { return nullptr; };
376
377
2/2
✓ Branch 41 → 16 taken 99 times.
✓ Branch 41 → 42 taken 102 times.
201 for (size_t i = 0; i < args.size(); i++) {
378
1/2
✓ Branch 16 → 17 taken 99 times.
✗ Branch 16 → 71 not taken.
99 const QualType &actualType = args.at(i).first;
379
1/2
✓ Branch 17 → 18 taken 99 times.
✗ Branch 17 → 71 not taken.
99 const QualType &expectedType = expectedArgTypes.at(i);
380
2/4
✓ Branch 19 → 20 taken 99 times.
✗ Branch 19 → 68 not taken.
✗ Branch 20 → 21 not taken.
✓ Branch 20 → 34 taken 99 times.
99 if (TypeMapping tm; !TypeMatcher::matchRequestedToCandidateType(expectedType, actualType, tm, resolverFct, false))
381
1/16
✗ Branch 21 → 22 not taken.
✗ Branch 21 → 66 not taken.
✗ Branch 22 → 23 not taken.
✗ Branch 22 → 61 not taken.
✗ Branch 23 → 24 not taken.
✗ Branch 23 → 59 not taken.
✗ Branch 24 → 25 not taken.
✗ Branch 24 → 57 not taken.
✗ Branch 25 → 26 not taken.
✗ Branch 25 → 55 not taken.
✗ Branch 26 → 27 not taken.
✗ Branch 26 → 53 not taken.
✗ Branch 27 → 28 not taken.
✗ Branch 27 → 53 not taken.
✓ Branch 36 → 37 taken 99 times.
✗ Branch 36 → 39 not taken.
99 SOFT_ERROR_BOOL(node->argLst->args.at(i), REFERENCED_UNDEFINED_FUNCTION,
382 "Expected " + expectedType.getName(false) + " but got " + actualType.getName(false))
383 }
384 102 return true;
385 102 }
386
387 26351 bool TypeChecker::visitMethodCall(FctCallNode *node, Scope *structScope) const {
388 26351 FctCallNode::FctCallData &data = node->data.at(manIdx);
389 26351 auto &[callType, isImported, templateTypes, thisType, args, callee, calleeParentScope, compTimeVal, hasCompTimeVal] = data;
390
391 // Traverse through structs - the first fragment is already looked up and the last one is the method name
392
2/2
✓ Branch 40 → 4 taken 7546 times.
✓ Branch 40 → 41 taken 26349 times.
33895 for (size_t i = 1; i < node->functionNameFragments.size() - 1; i++) {
393 7546 const std::string &identifier = node->functionNameFragments.at(i);
394
395 // Retrieve field entry
396 7546 SymbolTableEntry *fieldEntry = structScope->lookupStrict(identifier);
397
2/2
✓ Branch 8 → 9 taken 1 time.
✓ Branch 8 → 23 taken 7545 times.
7546 if (!fieldEntry) {
398
1/2
✓ Branch 9 → 10 taken 1 time.
✗ Branch 9 → 89 not taken.
1 std::stringstream errorMsg;
399
1/2
✓ Branch 10 → 11 taken 1 time.
✗ Branch 10 → 87 not taken.
1 errorMsg << "The type '";
400
3/6
✓ Branch 11 → 12 taken 1 time.
✗ Branch 11 → 82 not taken.
✓ Branch 12 → 13 taken 1 time.
✗ Branch 12 → 82 not taken.
✓ Branch 13 → 14 taken 1 time.
✗ Branch 13 → 80 not taken.
1 errorMsg << thisType.getBase().getName(false, true);
401
3/6
✓ Branch 15 → 16 taken 1 time.
✗ Branch 15 → 87 not taken.
✓ Branch 16 → 17 taken 1 time.
✗ Branch 16 → 87 not taken.
✓ Branch 17 → 18 taken 1 time.
✗ Branch 17 → 87 not taken.
1 errorMsg << "' does not have a member with the name '" << identifier << "'";
402
2/4
✓ Branch 18 → 19 taken 1 time.
✗ Branch 18 → 86 not taken.
✓ Branch 19 → 20 taken 1 time.
✗ Branch 19 → 84 not taken.
1 SOFT_ERROR_BOOL(node, ACCESS_TO_NON_EXISTING_MEMBER, errorMsg.str())
403 1 }
404
5/8
✓ Branch 23 → 24 taken 7545 times.
✗ Branch 23 → 91 not taken.
✓ Branch 24 → 25 taken 7545 times.
✗ Branch 24 → 91 not taken.
✓ Branch 25 → 26 taken 7545 times.
✗ Branch 25 → 90 not taken.
✓ Branch 26 → 27 taken 1 time.
✓ Branch 26 → 33 taken 7544 times.
7545 if (!fieldEntry->getQualType().getBase().isOneOf({TY_STRUCT, TY_INTERFACE}))
405
3/6
✓ Branch 27 → 28 taken 1 time.
✗ Branch 27 → 96 not taken.
✓ Branch 28 → 29 taken 1 time.
✗ Branch 28 → 94 not taken.
✓ Branch 29 → 30 taken 1 time.
✗ Branch 29 → 92 not taken.
1 SOFT_ERROR_BOOL(node, INVALID_MEMBER_ACCESS,
406 "Cannot call a method on '" + identifier + "', since it is no struct or interface")
407 7544 fieldEntry->used = true;
408
409 // Get struct type and scope
410 7544 thisType = fieldEntry->getQualType();
411
2/4
✓ Branch 34 → 35 taken 7544 times.
✗ Branch 34 → 98 not taken.
✓ Branch 35 → 36 taken 7544 times.
✗ Branch 35 → 98 not taken.
7544 structScope = thisType.getBase().getBodyScope();
412
1/2
✗ Branch 36 → 37 not taken.
✓ Branch 36 → 38 taken 7544 times.
7544 assert(structScope != nullptr);
413 }
414
415
1/2
✗ Branch 42 → 43 not taken.
✓ Branch 42 → 50 taken 26349 times.
26349 if (thisType.is(TY_INTERFACE))
416 SOFT_ERROR_BOOL(node, INVALID_MEMBER_ACCESS, "Cannot call a method on an interface")
417
418 // Map local arg types to imported types
419 26349 Scope *matchScope = calleeParentScope = structScope;
420
5/8
✓ Branch 50 → 51 taken 26349 times.
✗ Branch 50 → 106 not taken.
✓ Branch 51 → 52 taken 26349 times.
✗ Branch 51 → 106 not taken.
✓ Branch 52 → 53 taken 26349 times.
✗ Branch 52 → 106 not taken.
✓ Branch 58 → 54 taken 16069 times.
✓ Branch 58 → 59 taken 26349 times.
42418 for (QualType &argType : args | std::views::keys)
421
1/2
✓ Branch 55 → 56 taken 16069 times.
✗ Branch 55 → 105 not taken.
16069 argType = mapLocalTypeToImportedScopeType(calleeParentScope, argType);
422
423 // Map local template types to imported types
424
2/2
✓ Branch 73 → 61 taken 419 times.
✓ Branch 73 → 74 taken 26349 times.
53117 for (QualType &templateType : templateTypes)
425
1/2
✓ Branch 63 → 64 taken 419 times.
✗ Branch 63 → 107 not taken.
419 templateType = mapLocalTypeToImportedScopeType(calleeParentScope, templateType);
426
427 // 'this' type
428
1/2
✓ Branch 74 → 75 taken 26349 times.
✗ Branch 74 → 109 not taken.
26349 thisType = thisType.autoDeReference();
429
1/2
✓ Branch 75 → 76 taken 26349 times.
✗ Branch 75 → 110 not taken.
26349 thisType = mapLocalTypeToImportedScopeType(calleeParentScope, thisType);
430
431 // Retrieve function object
432 26349 const std::string &functionName = node->functionNameFragments.back();
433 26349 callee = FunctionManager::match(matchScope, functionName, thisType, args, templateTypes, false, node);
434
435 26349 return true;
436 }
437
438 326 std::any TypeChecker::visitArrayInitialization(ArrayInitializationNode *node) {
439
5/6
✓ Branch 2 → 3 taken 325 times.
✓ Branch 2 → 5 taken 1 time.
✗ Branch 4 → 5 not taken.
✓ Branch 4 → 6 taken 325 times.
✓ Branch 7 → 8 taken 1 time.
✓ Branch 7 → 18 taken 325 times.
326 if (!node->itemLst || node->itemLst->args.empty())
440
4/8
✓ Branch 10 → 11 taken 1 time.
✗ Branch 10 → 76 not taken.
✓ Branch 11 → 12 taken 1 time.
✗ Branch 11 → 74 not taken.
✓ Branch 14 → 15 taken 1 time.
✗ Branch 14 → 80 not taken.
✓ Branch 15 → 16 taken 1 time.
✗ Branch 15 → 80 not taken.
4 SOFT_ERROR_ER(node, ARRAY_SIZE_INVALID, "Array initializers must at least contain one value");
441 325 node->actualSize = node->itemLst->args.size();
442
443
1/2
✓ Branch 19 → 20 taken 325 times.
✗ Branch 19 → 105 not taken.
325 QualType actualItemType(TY_DYN);
444 // Check if all values have the same type
445
2/2
✓ Branch 63 → 22 taken 1453 times.
✓ Branch 63 → 64 taken 324 times.
2102 for (ExprNode *arg : node->itemLst->args) {
446
2/4
✓ Branch 24 → 25 taken 1453 times.
✗ Branch 24 → 83 not taken.
✓ Branch 25 → 26 taken 1453 times.
✗ Branch 25 → 81 not taken.
1453 const QualType itemType = std::any_cast<ExprResult>(visit(arg)).type;
447
2/8
✓ Branch 27 → 28 taken 1453 times.
✗ Branch 27 → 102 not taken.
✗ Branch 28 → 29 not taken.
✓ Branch 28 → 33 taken 1453 times.
✗ Branch 29 → 30 not taken.
✗ Branch 29 → 85 not taken.
✗ Branch 30 → 31 not taken.
✗ Branch 30 → 85 not taken.
1453 HANDLE_UNRESOLVED_TYPE_ER(itemType)
448
3/4
✓ Branch 33 → 34 taken 1453 times.
✗ Branch 33 → 102 not taken.
✓ Branch 34 → 35 taken 325 times.
✓ Branch 34 → 36 taken 1128 times.
1453 if (actualItemType.is(TY_DYN)) // Perform type inference
449 325 actualItemType = itemType;
450
3/4
✓ Branch 36 → 37 taken 1128 times.
✗ Branch 36 → 102 not taken.
✓ Branch 37 → 38 taken 1 time.
✓ Branch 37 → 53 taken 1127 times.
1128 else if (itemType != actualItemType) // Check if types are matching
451
8/16
✓ Branch 38 → 39 taken 1 time.
✗ Branch 38 → 99 not taken.
✓ Branch 39 → 40 taken 1 time.
✗ Branch 39 → 94 not taken.
✓ Branch 40 → 41 taken 1 time.
✗ Branch 40 → 92 not taken.
✓ Branch 41 → 42 taken 1 time.
✗ Branch 41 → 90 not taken.
✓ Branch 42 → 43 taken 1 time.
✗ Branch 42 → 88 not taken.
✓ Branch 43 → 44 taken 1 time.
✗ Branch 43 → 86 not taken.
✓ Branch 49 → 50 taken 1 time.
✗ Branch 49 → 101 not taken.
✓ Branch 50 → 51 taken 1 time.
✗ Branch 50 → 101 not taken.
2 SOFT_ERROR_ER(arg, ARRAY_ITEM_TYPE_NOT_MATCHING,
452 "All provided values have to be of the same data type. You provided " + actualItemType.getName(false) +
453 " and " + itemType.getName(false))
454 }
455
2/4
✓ Branch 64 → 65 taken 324 times.
✗ Branch 64 → 105 not taken.
✗ Branch 65 → 66 not taken.
✓ Branch 65 → 67 taken 324 times.
324 assert(!actualItemType.is(TY_DYN));
456
457
1/2
✓ Branch 67 → 68 taken 324 times.
✗ Branch 67 → 105 not taken.
324 const QualType arrayType = actualItemType.toArr(node, node->actualSize, true);
458
2/4
✓ Branch 68 → 69 taken 324 times.
✗ Branch 68 → 104 not taken.
✓ Branch 69 → 70 taken 324 times.
✗ Branch 69 → 104 not taken.
648 return ExprResult{node->setEvaluatedSymbolType(arrayType, manIdx)};
459 }
460
461 1312 std::any TypeChecker::visitStructInstantiation(StructInstantiationNode *node) {
462 // Retrieve struct name
463
1/2
✓ Branch 2 → 3 taken 1312 times.
✗ Branch 2 → 297 not taken.
1312 const auto [aliasedEntry, isAlias] = rootScope->symbolTable.lookupWithAliasResolution(node->fqStructName);
464
5/8
✓ Branch 5 → 6 taken 1 time.
✓ Branch 5 → 9 taken 1311 times.
✓ Branch 6 → 7 taken 1 time.
✗ Branch 6 → 297 not taken.
✓ Branch 7 → 8 taken 1 time.
✗ Branch 7 → 297 not taken.
✓ Branch 10 → 11 taken 1312 times.
✗ Branch 10 → 297 not taken.
1312 std::string structName = isAlias ? aliasedEntry->getQualType().getSubType() : node->fqStructName;
465
466 // Check if the struct type is generic
467
1/2
✓ Branch 11 → 12 taken 1312 times.
✗ Branch 11 → 295 not taken.
1312 const QualType *genericType = rootScope->lookupGenericTypeStrict(structName);
468
6/8
✓ Branch 12 → 13 taken 2 times.
✓ Branch 12 → 16 taken 1310 times.
✓ Branch 13 → 14 taken 2 times.
✗ Branch 13 → 295 not taken.
✓ Branch 14 → 15 taken 2 times.
✗ Branch 14 → 16 not taken.
✓ Branch 17 → 18 taken 2 times.
✓ Branch 17 → 23 taken 1310 times.
1312 if (genericType && typeMapping.contains(structName)) {
469
1/2
✓ Branch 18 → 19 taken 2 times.
✗ Branch 18 → 295 not taken.
2 const QualType &replacementType = typeMapping.at(structName);
470
2/4
✓ Branch 19 → 20 taken 2 times.
✗ Branch 19 → 295 not taken.
✓ Branch 20 → 21 taken 2 times.
✗ Branch 20 → 23 not taken.
2 if (replacementType.is(TY_STRUCT))
471
2/4
✓ Branch 21 → 22 taken 2 times.
✗ Branch 21 → 295 not taken.
✓ Branch 22 → 23 taken 2 times.
✗ Branch 22 → 295 not taken.
2 structName = replacementType.getSubType();
472 }
473
474 // Retrieve struct
475
1/2
✓ Branch 23 → 24 taken 1312 times.
✗ Branch 23 → 295 not taken.
1312 const NameRegistryEntry *registryEntry = sourceFile->getNameRegistryEntry(structName);
476
2/2
✓ Branch 24 → 25 taken 1 time.
✓ Branch 24 → 34 taken 1311 times.
1312 if (!registryEntry)
477
5/10
✓ Branch 25 → 26 taken 1 time.
✗ Branch 25 → 224 not taken.
✓ Branch 26 → 27 taken 1 time.
✗ Branch 26 → 222 not taken.
✓ Branch 27 → 28 taken 1 time.
✗ Branch 27 → 220 not taken.
✓ Branch 30 → 31 taken 1 time.
✗ Branch 30 → 226 not taken.
✓ Branch 31 → 32 taken 1 time.
✗ Branch 31 → 226 not taken.
2 SOFT_ERROR_ER(node, REFERENCED_UNDEFINED_STRUCT, "Cannot find struct '" + structName + "'")
478
2/4
✓ Branch 34 → 35 taken 1311 times.
✗ Branch 34 → 37 not taken.
✓ Branch 35 → 36 taken 1311 times.
✗ Branch 35 → 37 not taken.
1311 assert(registryEntry->targetEntry != nullptr && registryEntry->targetScope != nullptr);
479 1311 SymbolTableEntry *structEntry = registryEntry->targetEntry;
480
481 // Check visibility
482
9/12
✓ Branch 38 → 39 taken 1311 times.
✗ Branch 38 → 295 not taken.
✓ Branch 39 → 40 taken 1311 times.
✗ Branch 39 → 295 not taken.
✓ Branch 40 → 41 taken 98 times.
✓ Branch 40 → 44 taken 1213 times.
✓ Branch 41 → 42 taken 98 times.
✗ Branch 41 → 295 not taken.
✓ Branch 42 → 43 taken 1 time.
✓ Branch 42 → 44 taken 97 times.
✓ Branch 45 → 46 taken 1 time.
✓ Branch 45 → 55 taken 1310 times.
1311 if (!structEntry->getQualType().isPublic() && structEntry->scope->isImportedBy(currentScope))
483
5/10
✓ Branch 46 → 47 taken 1 time.
✗ Branch 46 → 231 not taken.
✓ Branch 47 → 48 taken 1 time.
✗ Branch 47 → 229 not taken.
✓ Branch 48 → 49 taken 1 time.
✗ Branch 48 → 227 not taken.
✓ Branch 51 → 52 taken 1 time.
✗ Branch 51 → 233 not taken.
✓ Branch 52 → 53 taken 1 time.
✗ Branch 52 → 233 not taken.
2 SOFT_ERROR_ER(node, INSUFFICIENT_VISIBILITY, "Struct '" + structName + "' has insufficient visibility")
484
485 // Get struct type
486
1/2
✓ Branch 55 → 56 taken 1310 times.
✗ Branch 55 → 295 not taken.
1310 QualType structType = structEntry->getQualType();
487
488 // Get the concrete template types
489 1310 QualTypeList concreteTemplateTypes;
490
2/2
✓ Branch 56 → 57 taken 1 time.
✓ Branch 56 → 76 taken 1309 times.
1310 if (isAlias) {
491 // Retrieve concrete template types from type alias
492
3/6
✓ Branch 57 → 58 taken 1 time.
✗ Branch 57 → 293 not taken.
✓ Branch 58 → 59 taken 1 time.
✗ Branch 58 → 293 not taken.
✓ Branch 59 → 60 taken 1 time.
✗ Branch 59 → 293 not taken.
1 concreteTemplateTypes = aliasedEntry->getQualType().getTemplateTypes();
493 // Check if the aliased type specified template types and the struct instantiation does
494
3/6
✓ Branch 61 → 62 taken 1 time.
✗ Branch 61 → 64 not taken.
✗ Branch 62 → 63 not taken.
✓ Branch 62 → 64 taken 1 time.
✗ Branch 65 → 66 not taken.
✓ Branch 65 → 76 taken 1 time.
1 if (!concreteTemplateTypes.empty() && node->templateTypeLst)
495 SOFT_ERROR_ER(node->templateTypeLst, ALIAS_WITH_TEMPLATE_LIST, "The aliased type already has a template list")
496 }
497
498
2/2
✓ Branch 76 → 77 taken 31 times.
✓ Branch 76 → 117 taken 1279 times.
1310 if (node->templateTypeLst) {
499
1/2
✓ Branch 78 → 79 taken 31 times.
✗ Branch 78 → 293 not taken.
31 concreteTemplateTypes.reserve(node->templateTypeLst->dataTypes.size());
500
2/2
✓ Branch 115 → 81 taken 57 times.
✓ Branch 115 → 116 taken 31 times.
119 for (DataTypeNode *dataType : node->templateTypeLst->dataTypes) {
501
2/4
✓ Branch 83 → 84 taken 57 times.
✗ Branch 83 → 243 not taken.
✓ Branch 84 → 85 taken 57 times.
✗ Branch 84 → 241 not taken.
57 auto concreteType = std::any_cast<QualType>(visit(dataType));
502
2/8
✓ Branch 86 → 87 taken 57 times.
✗ Branch 86 → 252 not taken.
✗ Branch 87 → 88 not taken.
✓ Branch 87 → 92 taken 57 times.
✗ Branch 88 → 89 not taken.
✗ Branch 88 → 244 not taken.
✗ Branch 89 → 90 not taken.
✗ Branch 89 → 244 not taken.
57 HANDLE_UNRESOLVED_TYPE_ER(concreteType)
503 // Check if generic type
504
2/4
✓ Branch 92 → 93 taken 57 times.
✗ Branch 92 → 252 not taken.
✗ Branch 93 → 94 not taken.
✓ Branch 93 → 104 taken 57 times.
57 if (concreteType.is(TY_GENERIC))
505 SOFT_ERROR_ER(dataType, EXPECTED_NON_GENERIC_TYPE, "Struct instantiations may only take concrete template types")
506
1/2
✓ Branch 104 → 105 taken 57 times.
✗ Branch 104 → 252 not taken.
57 concreteTemplateTypes.push_back(concreteType);
507 }
508 }
509
510 // Get the struct instance
511
2/4
✓ Branch 117 → 118 taken 1310 times.
✗ Branch 117 → 293 not taken.
✓ Branch 118 → 119 taken 1310 times.
✗ Branch 118 → 293 not taken.
1310 Struct *spiceStruct = node->instantiatedStructs.at(manIdx) = structType.getStructAndAdjustType(node, concreteTemplateTypes);
512
1/2
✗ Branch 119 → 120 not taken.
✓ Branch 119 → 131 taken 1310 times.
1310 if (!spiceStruct)
513 SOFT_ERROR_ER(node, REFERENCED_UNDEFINED_STRUCT,
514 "Struct '" + Struct::getSignature(structName, concreteTemplateTypes) + "' could not be found")
515
516 // Struct instantiation for an inheriting struct is forbidden, because the vtable needs to be initialized and this is done in
517 // the ctor of the struct, which is never called in case of struct instantiation
518
2/2
✓ Branch 132 → 133 taken 1 time.
✓ Branch 132 → 143 taken 1309 times.
1310 if (!spiceStruct->interfaceTypes.empty())
519
4/8
✓ Branch 135 → 136 taken 1 time.
✗ Branch 135 → 266 not taken.
✓ Branch 136 → 137 taken 1 time.
✗ Branch 136 → 264 not taken.
✓ Branch 139 → 140 taken 1 time.
✗ Branch 139 → 270 not taken.
✓ Branch 140 → 141 taken 1 time.
✗ Branch 140 → 270 not taken.
4 SOFT_ERROR_ER(node, INVALID_STRUCT_INSTANTIATION, "Struct instantiations for inheriting structs are forbidden")
520
521 // Check if the number of fields matches
522
2/2
✓ Branch 143 → 144 taken 1284 times.
✓ Branch 143 → 194 taken 25 times.
1309 if (node->fieldLst) { // Check if any fields are passed. Empty braces are also allowed
523
2/2
✓ Branch 146 → 147 taken 1 time.
✓ Branch 146 → 157 taken 1283 times.
1284 if (spiceStruct->fieldTypes.size() != node->fieldLst->args.size())
524
4/8
✓ Branch 149 → 150 taken 1 time.
✗ Branch 149 → 273 not taken.
✓ Branch 150 → 151 taken 1 time.
✗ Branch 150 → 271 not taken.
✓ Branch 153 → 154 taken 1 time.
✗ Branch 153 → 277 not taken.
✓ Branch 154 → 155 taken 1 time.
✗ Branch 154 → 277 not taken.
4 SOFT_ERROR_ER(node->fieldLst, NUMBER_OF_FIELDS_NOT_MATCHING,
525 "You've passed too less/many field values. Pass either none or all of them")
526
527 // Check if the field types are matching
528 1283 const size_t fieldCount = spiceStruct->fieldTypes.size();
529
1/2
✓ Branch 158 → 159 taken 1283 times.
✗ Branch 158 → 293 not taken.
1283 const size_t explicitFieldsStartIdx = spiceStruct->scope->getFieldCount() - fieldCount;
530
2/2
✓ Branch 193 → 160 taken 1736 times.
✓ Branch 193 → 206 taken 1282 times.
3018 for (size_t i = 0; i < node->fieldLst->args.size(); i++) {
531 // Get actual type
532
1/2
✓ Branch 160 → 161 taken 1736 times.
✗ Branch 160 → 284 not taken.
1736 ExprNode *assignExpr = node->fieldLst->args.at(i);
533
2/4
✓ Branch 161 → 162 taken 1736 times.
✗ Branch 161 → 280 not taken.
✓ Branch 162 → 163 taken 1736 times.
✗ Branch 162 → 278 not taken.
1736 auto fieldResult = std::any_cast<ExprResult>(visit(assignExpr));
534
2/8
✓ Branch 164 → 165 taken 1736 times.
✗ Branch 164 → 284 not taken.
✗ Branch 165 → 166 not taken.
✓ Branch 165 → 170 taken 1736 times.
✗ Branch 166 → 167 not taken.
✗ Branch 166 → 281 not taken.
✗ Branch 167 → 168 not taken.
✗ Branch 167 → 281 not taken.
1736 HANDLE_UNRESOLVED_TYPE_ER(fieldResult.type)
535 // Get expected type
536
1/2
✗ Branch 170 → 171 not taken.
✓ Branch 170 → 172 taken 1736 times.
1736 SymbolTableEntry *expectedField = spiceStruct->scope->lookupField(explicitFieldsStartIdx + i);
537
1/2
✗ Branch 175 → 176 not taken.
✓ Branch 175 → 177 taken 1736 times.
1736 assert(expectedField != nullptr);
538
1/2
✓ Branch 177 → 178 taken 1736 times.
✗ Branch 177 → 284 not taken.
1736 const ExprResult expected = {expectedField->getQualType(), expectedField};
539
1/2
✓ Branch 178 → 179 taken 1736 times.
✗ Branch 178 → 284 not taken.
1736 const bool rhsIsImmediate = assignExpr->hasCompileTimeValue(manIdx);
540
541 // Capture anonymous info up front: getFieldAssignResultType may delete the anonymous entry (temp stealing
542 // in performStructAssign), turning fieldResult.entry into a dangling pointer.
543
4/4
✓ Branch 179 → 180 taken 385 times.
✓ Branch 179 → 182 taken 1351 times.
✓ Branch 180 → 181 taken 7 times.
✓ Branch 180 → 182 taken 378 times.
1736 const bool rhsIsAnonymous = fieldResult.entry != nullptr && fieldResult.entry->anonymous;
544
3/4
✓ Branch 183 → 184 taken 7 times.
✓ Branch 183 → 185 taken 1729 times.
✓ Branch 184 → 186 taken 7 times.
✗ Branch 184 → 284 not taken.
1736 const std::string rhsEntryName = rhsIsAnonymous ? fieldResult.entry->name : std::string();
545
546 // Check if actual type matches expected type
547
2/2
✓ Branch 186 → 187 taken 1735 times.
✓ Branch 186 → 282 taken 1 time.
1736 (void)opRuleManager.getFieldAssignResultType(assignExpr, expected, fieldResult, rhsIsImmediate, true);
548
549 // If there is an anonymous entry attached (e.g. for struct instantiation), delete it.
550 // Safe to call even if performStructAssign already deleted it: map::erase by key is a no-op when absent.
551
2/2
✓ Branch 187 → 188 taken 7 times.
✓ Branch 187 → 190 taken 1728 times.
1735 if (rhsIsAnonymous) {
552
1/2
✓ Branch 188 → 189 taken 7 times.
✗ Branch 188 → 282 not taken.
7 currentScope->symbolTable.deleteAnonymous(rhsEntryName);
553 7 fieldResult.entry = nullptr;
554 }
555 1736 }
556 } else {
557
2/4
✓ Branch 194 → 195 taken 25 times.
✗ Branch 194 → 293 not taken.
✗ Branch 195 → 196 not taken.
✓ Branch 195 → 206 taken 25 times.
97 if (std::ranges::any_of(spiceStruct->fieldTypes, [](const QualType &fieldType) { return fieldType.isRef(); }))
558 SOFT_ERROR_ER(node, REFERENCE_WITHOUT_INITIALIZER,
559 "The struct takes at least one reference field. You need to instantiate it with all fields.")
560 }
561
562 // Update type of struct entry
563
1/2
✓ Branch 206 → 207 taken 1307 times.
✗ Branch 206 → 293 not taken.
1307 structEntry->updateType(structType, true);
564
565 // Add anonymous symbol to keep track of dtor call, if non-trivially destructible
566 1307 SymbolTableEntry *anonymousEntry = nullptr;
567
3/4
✓ Branch 207 → 208 taken 1307 times.
✗ Branch 207 → 293 not taken.
✓ Branch 208 → 209 taken 97 times.
✓ Branch 208 → 211 taken 1210 times.
1307 if (!structType.isTriviallyDestructible(node))
568
1/2
✓ Branch 209 → 210 taken 97 times.
✗ Branch 209 → 293 not taken.
97 anonymousEntry = currentScope->symbolTable.insertAnonymous(structType, node);
569
570 // Remove public qualifier to not have public local variables
571 1307 structType.getQualifiers().isPublic = false;
572
573
2/4
✓ Branch 212 → 213 taken 1307 times.
✗ Branch 212 → 292 not taken.
✓ Branch 213 → 214 taken 1307 times.
✗ Branch 213 → 292 not taken.
2614 return ExprResult{node->setEvaluatedSymbolType(structType, manIdx), anonymousEntry};
574 1313 }
575
576 22 std::any TypeChecker::visitLambdaFunc(LambdaFuncNode *node) {
577 // Check if all control paths in the lambda body return
578 22 bool returnsOnAllControlPaths = true;
579
3/4
✓ Branch 2 → 3 taken 22 times.
✗ Branch 2 → 194 not taken.
✓ Branch 3 → 4 taken 1 time.
✓ Branch 3 → 14 taken 21 times.
22 if (!node->returnsOnAllControlPaths(&returnsOnAllControlPaths, manIdx))
580
4/8
✓ Branch 6 → 7 taken 1 time.
✗ Branch 6 → 120 not taken.
✓ Branch 7 → 8 taken 1 time.
✗ Branch 7 → 118 not taken.
✓ Branch 10 → 11 taken 1 time.
✗ Branch 10 → 124 not taken.
✓ Branch 11 → 12 taken 1 time.
✗ Branch 11 → 124 not taken.
4 SOFT_ERROR_ER(node, MISSING_RETURN_STMT, "Not all control paths of this lambda function have a return statement")
581
582 // Change to function scope
583
2/4
✓ Branch 14 → 15 taken 21 times.
✗ Branch 14 → 127 not taken.
✓ Branch 15 → 16 taken 21 times.
✗ Branch 15 → 125 not taken.
21 Scope *bodyScope = currentScope->getChildScope(node->getScopeId());
584
1/2
✓ Branch 17 → 18 taken 21 times.
✗ Branch 17 → 128 not taken.
21 ScopeHandle scopeHandle(this, bodyScope, ScopeType::LAMBDA_BODY);
585
586 // Visit return type
587
2/4
✓ Branch 18 → 19 taken 21 times.
✗ Branch 18 → 131 not taken.
✓ Branch 19 → 20 taken 21 times.
✗ Branch 19 → 129 not taken.
21 const auto returnType = std::any_cast<QualType>(visit(node->returnType));
588
2/8
✓ Branch 21 → 22 taken 21 times.
✗ Branch 21 → 192 not taken.
✗ Branch 22 → 23 not taken.
✓ Branch 22 → 27 taken 21 times.
✗ Branch 23 → 24 not taken.
✗ Branch 23 → 132 not taken.
✗ Branch 24 → 25 not taken.
✗ Branch 24 → 132 not taken.
21 HANDLE_UNRESOLVED_TYPE_ER(returnType)
589
3/4
✓ Branch 27 → 28 taken 21 times.
✗ Branch 27 → 192 not taken.
✓ Branch 28 → 29 taken 1 time.
✓ Branch 28 → 39 taken 20 times.
21 if (returnType.is(TY_DYN))
590
4/8
✓ Branch 31 → 32 taken 1 time.
✗ Branch 31 → 135 not taken.
✓ Branch 32 → 33 taken 1 time.
✗ Branch 32 → 133 not taken.
✓ Branch 35 → 36 taken 1 time.
✗ Branch 35 → 139 not taken.
✓ Branch 36 → 37 taken 1 time.
✗ Branch 36 → 139 not taken.
4 SOFT_ERROR_ER(node, UNEXPECTED_DYN_TYPE, "Dyn return types are not allowed")
591
592 // Set the type of the result variable
593
1/2
✓ Branch 41 → 42 taken 20 times.
✗ Branch 41 → 142 not taken.
60 SymbolTableEntry *resultVarEntry = currentScope->lookupStrict(RETURN_VARIABLE_NAME);
594
1/2
✗ Branch 47 → 48 not taken.
✓ Branch 47 → 49 taken 20 times.
20 assert(resultVarEntry != nullptr);
595
1/2
✓ Branch 49 → 50 taken 20 times.
✗ Branch 49 → 192 not taken.
20 resultVarEntry->updateType(returnType, true);
596 20 resultVarEntry->used = true;
597
598 // Visit parameters
599 20 QualTypeList paramTypes;
600 20 ParamList paramList;
601
2/2
✓ Branch 50 → 51 taken 15 times.
✓ Branch 50 → 80 taken 5 times.
20 if (node->hasParams) {
602 // Visit param list to retrieve the param names
603
2/4
✓ Branch 51 → 52 taken 15 times.
✗ Branch 51 → 148 not taken.
✓ Branch 52 → 53 taken 15 times.
✗ Branch 52 → 146 not taken.
15 auto namedParamList = std::any_cast<NamedParamList>(visit(node->paramLst));
604
2/2
✓ Branch 77 → 56 taken 23 times.
✓ Branch 77 → 78 taken 15 times.
53 for (const auto &[name, qualType, isOptional] : namedParamList) {
605
1/2
✗ Branch 58 → 59 not taken.
✓ Branch 58 → 66 taken 23 times.
23 if (isOptional)
606 softError(node, LAMBDA_WITH_OPTIONAL_PARAMS, "Lambdas cannot have optional parameters");
607
608
1/2
✓ Branch 66 → 67 taken 23 times.
✗ Branch 66 → 156 not taken.
23 paramTypes.push_back(qualType);
609
1/2
✓ Branch 67 → 68 taken 23 times.
✗ Branch 67 → 155 not taken.
23 paramList.push_back({qualType, isOptional});
610 }
611 15 }
612
613 // Visit lambda body
614
2/2
✓ Branch 80 → 81 taken 19 times.
✓ Branch 80 → 160 taken 1 time.
20 visit(node->body);
615
616 // Leave function body scope
617
1/2
✓ Branch 82 → 83 taken 19 times.
✗ Branch 82 → 188 not taken.
19 scopeHandle.leaveScopeEarly();
618
619 // Prepare type of function
620
1/2
✓ Branch 83 → 84 taken 19 times.
✗ Branch 83 → 161 not taken.
19 const QualType functionType = QualType(TY_FUNCTION)
621
1/2
✓ Branch 84 → 85 taken 19 times.
✗ Branch 84 → 161 not taken.
19 .getWithFunctionParamAndReturnTypes(returnType, paramTypes)
622
1/2
✓ Branch 86 → 87 taken 19 times.
✗ Branch 86 → 161 not taken.
19 .getWithLambdaCaptures(!bodyScope->symbolTable.captures.empty());
623
624 // Create function object
625
2/4
✓ Branch 87 → 88 taken 19 times.
✗ Branch 87 → 165 not taken.
✓ Branch 88 → 89 taken 19 times.
✗ Branch 88 → 163 not taken.
19 const std::string fctName = "lambda." + node->codeLoc.toPrettyLineAndColumn();
626
5/10
✓ Branch 91 → 92 taken 19 times.
✗ Branch 91 → 174 not taken.
✓ Branch 92 → 93 taken 19 times.
✗ Branch 92 → 171 not taken.
✓ Branch 93 → 94 taken 19 times.
✗ Branch 93 → 170 not taken.
✓ Branch 94 → 95 taken 19 times.
✗ Branch 94 → 168 not taken.
✓ Branch 95 → 96 taken 19 times.
✗ Branch 95 → 166 not taken.
19 node->manifestations.at(manIdx) = Function(fctName, nullptr, QualType(TY_DYN), returnType, paramList, {}, node);
627
1/2
✓ Branch 101 → 102 taken 19 times.
✗ Branch 101 → 186 not taken.
19 node->manifestations.at(manIdx).bodyScope = bodyScope;
628
3/6
✓ Branch 102 → 103 taken 19 times.
✗ Branch 102 → 183 not taken.
✓ Branch 103 → 104 taken 19 times.
✗ Branch 103 → 181 not taken.
✓ Branch 104 → 105 taken 19 times.
✗ Branch 104 → 179 not taken.
19 node->manifestations.at(manIdx).mangleSuffix = "." + std::to_string(manIdx);
629
630 // Check special requirements if this is an async lambda
631
1/2
✓ Branch 108 → 109 taken 19 times.
✗ Branch 108 → 186 not taken.
19 (void)checkAsyncLambdaCaptureRules(node, node->lambdaAttr);
632
633
2/4
✓ Branch 109 → 110 taken 19 times.
✗ Branch 109 → 185 not taken.
✓ Branch 110 → 111 taken 19 times.
✗ Branch 110 → 185 not taken.
19 return ExprResult{node->setEvaluatedSymbolType(functionType, manIdx)};
634 23 }
635
636 43 std::any TypeChecker::visitLambdaProc(LambdaProcNode *node) {
637 // Mark unreachable statements
638 43 bool doSetPredecessorsUnreachable = true;
639
1/2
✓ Branch 2 → 3 taken 43 times.
✗ Branch 2 → 131 not taken.
43 node->returnsOnAllControlPaths(&doSetPredecessorsUnreachable, manIdx);
640
641 // Change to function scope
642
2/4
✓ Branch 3 → 4 taken 43 times.
✗ Branch 3 → 79 not taken.
✓ Branch 4 → 5 taken 43 times.
✗ Branch 4 → 77 not taken.
43 Scope *bodyScope = currentScope->getChildScope(node->getScopeId());
643
1/2
✓ Branch 6 → 7 taken 43 times.
✗ Branch 6 → 80 not taken.
43 ScopeHandle scopeHandle(this, bodyScope, ScopeType::LAMBDA_BODY);
644
645 // Visit parameters
646 43 QualTypeList paramTypes;
647 43 ParamList paramList;
648
2/2
✓ Branch 7 → 8 taken 28 times.
✓ Branch 7 → 37 taken 15 times.
43 if (node->hasParams) {
649 // Visit param list to retrieve the param names
650
2/4
✓ Branch 8 → 9 taken 28 times.
✗ Branch 8 → 83 not taken.
✓ Branch 9 → 10 taken 28 times.
✗ Branch 9 → 81 not taken.
28 auto namedParamList = std::any_cast<NamedParamList>(visit(node->paramLst));
651
2/2
✓ Branch 34 → 13 taken 38 times.
✓ Branch 34 → 35 taken 28 times.
94 for (const auto &[_, qualType, isOptional] : namedParamList) {
652
2/2
✓ Branch 15 → 16 taken 1 time.
✓ Branch 15 → 23 taken 37 times.
38 if (isOptional)
653
2/4
✓ Branch 18 → 19 taken 1 time.
✗ Branch 18 → 86 not taken.
✓ Branch 19 → 20 taken 1 time.
✗ Branch 19 → 84 not taken.
2 softError(node, LAMBDA_WITH_OPTIONAL_PARAMS, "Lambdas cannot have optional parameters");
654
655
1/2
✓ Branch 23 → 24 taken 38 times.
✗ Branch 23 → 91 not taken.
38 paramTypes.push_back(qualType);
656
1/2
✓ Branch 24 → 25 taken 38 times.
✗ Branch 24 → 90 not taken.
38 paramList.push_back({qualType, isOptional});
657 }
658 28 }
659
660 // Visit lambda body
661
1/2
✓ Branch 37 → 38 taken 43 times.
✗ Branch 37 → 95 not taken.
43 visit(node->body);
662
663 // Leave function body scope
664
1/2
✓ Branch 39 → 40 taken 43 times.
✗ Branch 39 → 125 not taken.
43 scopeHandle.leaveScopeEarly();
665
666 // Prepare type of function
667
1/2
✓ Branch 40 → 41 taken 43 times.
✗ Branch 40 → 97 not taken.
43 const QualType functionType = QualType(TY_PROCEDURE)
668
2/4
✓ Branch 41 → 42 taken 43 times.
✗ Branch 41 → 96 not taken.
✓ Branch 42 → 43 taken 43 times.
✗ Branch 42 → 96 not taken.
43 .getWithFunctionParamAndReturnTypes(QualType(TY_DYN), paramTypes)
669
1/2
✓ Branch 44 → 45 taken 43 times.
✗ Branch 44 → 96 not taken.
43 .getWithLambdaCaptures(!bodyScope->symbolTable.captures.empty());
670
671 // Create function object
672
2/4
✓ Branch 45 → 46 taken 43 times.
✗ Branch 45 → 101 not taken.
✓ Branch 46 → 47 taken 43 times.
✗ Branch 46 → 99 not taken.
43 const std::string fctName = "lambda." + node->codeLoc.toPrettyLineAndColumn();
673
6/12
✓ Branch 49 → 50 taken 43 times.
✗ Branch 49 → 111 not taken.
✓ Branch 50 → 51 taken 43 times.
✗ Branch 50 → 108 not taken.
✓ Branch 51 → 52 taken 43 times.
✗ Branch 51 → 107 not taken.
✓ Branch 52 → 53 taken 43 times.
✗ Branch 52 → 106 not taken.
✓ Branch 53 → 54 taken 43 times.
✗ Branch 53 → 104 not taken.
✓ Branch 54 → 55 taken 43 times.
✗ Branch 54 → 102 not taken.
43 node->manifestations.at(manIdx) = Function(fctName, nullptr, QualType(TY_DYN), QualType(TY_DYN), paramList, {}, node);
674
1/2
✓ Branch 60 → 61 taken 43 times.
✗ Branch 60 → 123 not taken.
43 node->manifestations.at(manIdx).bodyScope = bodyScope;
675
3/6
✓ Branch 61 → 62 taken 43 times.
✗ Branch 61 → 120 not taken.
✓ Branch 62 → 63 taken 43 times.
✗ Branch 62 → 118 not taken.
✓ Branch 63 → 64 taken 43 times.
✗ Branch 63 → 116 not taken.
43 node->manifestations.at(manIdx).mangleSuffix = "." + std::to_string(manIdx);
676
677 // Check special requirements if this is an async lambda
678
1/2
✓ Branch 67 → 68 taken 43 times.
✗ Branch 67 → 123 not taken.
43 (void)checkAsyncLambdaCaptureRules(node, node->lambdaAttr);
679
680
2/4
✓ Branch 68 → 69 taken 43 times.
✗ Branch 68 → 122 not taken.
✓ Branch 69 → 70 taken 43 times.
✗ Branch 69 → 122 not taken.
86 return ExprResult{node->setEvaluatedSymbolType(functionType, manIdx)};
681 43 }
682
683 1 std::any TypeChecker::visitLambdaExpr(LambdaExprNode *node) {
684 // Change to function scope
685
2/4
✓ Branch 2 → 3 taken 1 time.
✗ Branch 2 → 98 not taken.
✓ Branch 3 → 4 taken 1 time.
✗ Branch 3 → 96 not taken.
1 Scope *bodyScope = currentScope->getChildScope(node->getScopeId());
686
1/2
✓ Branch 5 → 6 taken 1 time.
✗ Branch 5 → 99 not taken.
1 ScopeHandle scopeHandle(this, bodyScope, ScopeType::LAMBDA_BODY);
687
688 // Visit parameters
689 1 QualTypeList paramTypes;
690 1 ParamList paramList;
691
1/2
✓ Branch 6 → 7 taken 1 time.
✗ Branch 6 → 36 not taken.
1 if (node->hasParams) {
692 // Visit param list to retrieve the param names
693
2/4
✓ Branch 7 → 8 taken 1 time.
✗ Branch 7 → 102 not taken.
✓ Branch 8 → 9 taken 1 time.
✗ Branch 8 → 100 not taken.
1 auto namedParamList = std::any_cast<NamedParamList>(visit(node->paramLst));
694
2/2
✓ Branch 33 → 12 taken 2 times.
✓ Branch 33 → 34 taken 1 time.
4 for (const NamedParam &param : namedParamList) {
695
1/2
✗ Branch 14 → 15 not taken.
✓ Branch 14 → 22 taken 2 times.
2 if (param.isOptional)
696 softError(node, LAMBDA_WITH_OPTIONAL_PARAMS, "Lambdas cannot have optional parameters");
697
698
1/2
✓ Branch 22 → 23 taken 2 times.
✗ Branch 22 → 110 not taken.
2 paramTypes.push_back(param.qualType);
699
1/2
✓ Branch 23 → 24 taken 2 times.
✗ Branch 23 → 109 not taken.
2 paramList.push_back({param.qualType, param.isOptional});
700 }
701 1 }
702
703 // Visit lambda expression
704
2/4
✓ Branch 36 → 37 taken 1 time.
✗ Branch 36 → 116 not taken.
✓ Branch 37 → 38 taken 1 time.
✗ Branch 37 → 114 not taken.
1 const QualType returnType = std::any_cast<ExprResult>(visit(node->lambdaExpr)).type;
705
2/8
✓ Branch 39 → 40 taken 1 time.
✗ Branch 39 → 153 not taken.
✗ Branch 40 → 41 not taken.
✓ Branch 40 → 45 taken 1 time.
✗ Branch 41 → 42 not taken.
✗ Branch 41 → 118 not taken.
✗ Branch 42 → 43 not taken.
✗ Branch 42 → 118 not taken.
1 HANDLE_UNRESOLVED_TYPE_ER(returnType)
706
2/4
✓ Branch 45 → 46 taken 1 time.
✗ Branch 45 → 153 not taken.
✗ Branch 46 → 47 not taken.
✓ Branch 46 → 57 taken 1 time.
1 if (returnType.is(TY_DYN))
707 SOFT_ERROR_ER(node, UNEXPECTED_DYN_TYPE, "Dyn return types are not allowed")
708
709 // Leave function body scope
710
1/2
✓ Branch 57 → 58 taken 1 time.
✗ Branch 57 → 153 not taken.
1 scopeHandle.leaveScopeEarly();
711
712 // Prepare type of function
713
2/4
✓ Branch 58 → 59 taken 1 time.
✗ Branch 58 → 153 not taken.
✗ Branch 59 → 60 not taken.
✓ Branch 59 → 61 taken 1 time.
1 const SuperType superType = returnType.is(TY_DYN) ? TY_PROCEDURE : TY_FUNCTION;
714
1/2
✓ Branch 62 → 63 taken 1 time.
✗ Branch 62 → 126 not taken.
1 const QualType functionType = QualType(superType)
715
1/2
✓ Branch 63 → 64 taken 1 time.
✗ Branch 63 → 126 not taken.
1 .getWithFunctionParamAndReturnTypes(returnType, paramTypes)
716
1/2
✓ Branch 65 → 66 taken 1 time.
✗ Branch 65 → 126 not taken.
1 .getWithLambdaCaptures(!bodyScope->symbolTable.captures.empty());
717
718 // Create function object
719
2/4
✓ Branch 66 → 67 taken 1 time.
✗ Branch 66 → 130 not taken.
✓ Branch 67 → 68 taken 1 time.
✗ Branch 67 → 128 not taken.
1 const std::string fctName = "lambda." + node->codeLoc.toPrettyLineAndColumn();
720
5/10
✓ Branch 70 → 71 taken 1 time.
✗ Branch 70 → 139 not taken.
✓ Branch 71 → 72 taken 1 time.
✗ Branch 71 → 136 not taken.
✓ Branch 72 → 73 taken 1 time.
✗ Branch 72 → 135 not taken.
✓ Branch 73 → 74 taken 1 time.
✗ Branch 73 → 133 not taken.
✓ Branch 74 → 75 taken 1 time.
✗ Branch 74 → 131 not taken.
1 node->manifestations.at(manIdx) = Function(fctName, nullptr, QualType(TY_DYN), returnType, paramList, {}, node);
721
1/2
✓ Branch 80 → 81 taken 1 time.
✗ Branch 80 → 151 not taken.
1 node->manifestations.at(manIdx).bodyScope = bodyScope;
722
3/6
✓ Branch 81 → 82 taken 1 time.
✗ Branch 81 → 148 not taken.
✓ Branch 82 → 83 taken 1 time.
✗ Branch 82 → 146 not taken.
✓ Branch 83 → 84 taken 1 time.
✗ Branch 83 → 144 not taken.
1 node->manifestations.at(manIdx).mangleSuffix = "." + std::to_string(manIdx);
723
724
2/4
✓ Branch 87 → 88 taken 1 time.
✗ Branch 87 → 150 not taken.
✓ Branch 88 → 89 taken 1 time.
✗ Branch 88 → 150 not taken.
1 return ExprResult{node->setEvaluatedSymbolType(functionType, manIdx)};
725 1 }
726
727 } // namespace spice::compiler
728