| Line | Branch | Exec | Source |
|---|---|---|---|
| 1 | // Copyright (c) 2021-2025 ChilliBits. All rights reserved. | ||
| 2 | |||
| 3 | #include "TypeChecker.h" | ||
| 4 | |||
| 5 | #include <SourceFile.h> | ||
| 6 | #include <global/GlobalResourceManager.h> | ||
| 7 | #include <symboltablebuilder/ScopeHandle.h> | ||
| 8 | #include <typechecker/MacroDefs.h> | ||
| 9 | |||
| 10 | namespace spice::compiler { | ||
| 11 | |||
| 12 | 2148 | std::any TypeChecker::visitUnsafeBlock(UnsafeBlockNode *node) { | |
| 13 | // Change to unsafe block body scope | ||
| 14 |
2/4✓ Branch 2 → 3 taken 2148 times.
✗ Branch 2 → 13 not taken.
✓ Branch 3 → 4 taken 2148 times.
✗ Branch 3 → 11 not taken.
|
2148 | ScopeHandle scopeHandle(this, node->getScopeId(), ScopeType::UNSAFE_BODY); |
| 15 | |||
| 16 | // Visit body | ||
| 17 |
1/2✓ Branch 5 → 6 taken 2148 times.
✗ Branch 5 → 15 not taken.
|
2148 | visit(node->body); |
| 18 | |||
| 19 |
1/2✓ Branch 7 → 8 taken 2148 times.
✗ Branch 7 → 16 not taken.
|
4296 | return nullptr; |
| 20 | 2148 | } | |
| 21 | |||
| 22 | 1274 | std::any TypeChecker::visitForLoop(ForLoopNode *node) { | |
| 23 | // Change to for body scope | ||
| 24 |
2/4✓ Branch 2 → 3 taken 1274 times.
✗ Branch 2 → 37 not taken.
✓ Branch 3 → 4 taken 1274 times.
✗ Branch 3 → 35 not taken.
|
1274 | ScopeHandle scopeHandle(this, node->getScopeId(), ScopeType::FOR_BODY); |
| 25 | |||
| 26 | // Visit loop variable declaration | ||
| 27 |
1/2✓ Branch 5 → 6 taken 1274 times.
✗ Branch 5 → 39 not taken.
|
1274 | visit(node->initDecl); |
| 28 | |||
| 29 | // Visit condition | ||
| 30 |
2/4✓ Branch 7 → 8 taken 1274 times.
✗ Branch 7 → 42 not taken.
✓ Branch 8 → 9 taken 1274 times.
✗ Branch 8 → 40 not taken.
|
1274 | const QualType conditionType = std::any_cast<ExprResult>(visit(node->condAssign)).type; |
| 31 |
2/6✓ Branch 10 → 11 taken 1274 times.
✗ Branch 10 → 55 not taken.
✗ Branch 11 → 12 not taken.
✓ Branch 11 → 14 taken 1274 times.
✗ Branch 12 → 13 not taken.
✗ Branch 12 → 44 not taken.
|
1274 | HANDLE_UNRESOLVED_TYPE_PTR(conditionType) |
| 32 | // Check if condition evaluates to bool | ||
| 33 |
3/4✓ Branch 14 → 15 taken 1274 times.
✗ Branch 14 → 55 not taken.
✓ Branch 15 → 16 taken 1 time.
✓ Branch 15 → 26 taken 1273 times.
|
1274 | if (!conditionType.is(TY_BOOL)) |
| 34 |
4/8✓ Branch 18 → 19 taken 1 time.
✗ Branch 18 → 47 not taken.
✓ Branch 19 → 20 taken 1 time.
✗ Branch 19 → 45 not taken.
✓ Branch 22 → 23 taken 1 time.
✗ Branch 22 → 51 not taken.
✓ Branch 23 → 24 taken 1 time.
✗ Branch 23 → 51 not taken.
|
3 | SOFT_ERROR_ER(node->condAssign, CONDITION_MUST_BE_BOOL, "For loop condition must be of type bool") |
| 35 | |||
| 36 | // Visit incrementer | ||
| 37 |
1/2✓ Branch 26 → 27 taken 1273 times.
✗ Branch 26 → 52 not taken.
|
1273 | visit(node->incAssign); |
| 38 | |||
| 39 | // Visit body | ||
| 40 |
1/2✓ Branch 28 → 29 taken 1273 times.
✗ Branch 28 → 53 not taken.
|
1273 | visit(node->body); |
| 41 | |||
| 42 |
1/2✓ Branch 30 → 31 taken 1273 times.
✗ Branch 30 → 54 not taken.
|
1273 | return nullptr; |
| 43 | 1274 | } | |
| 44 | |||
| 45 | 116 | std::any TypeChecker::visitForeachLoop(ForeachLoopNode *node) { | |
| 46 | // Visit iterator assignment | ||
| 47 | 116 | AssignExprNode *iteratorNode = node->iteratorAssign; | |
| 48 |
2/4✓ Branch 2 → 3 taken 116 times.
✗ Branch 2 → 202 not taken.
✓ Branch 3 → 4 taken 116 times.
✗ Branch 3 → 200 not taken.
|
116 | QualType iteratorOrIterableType = std::any_cast<ExprResult>(visit(iteratorNode)).type; |
| 49 |
2/6✓ Branch 5 → 6 taken 116 times.
✗ Branch 5 → 346 not taken.
✗ Branch 6 → 7 not taken.
✓ Branch 6 → 9 taken 116 times.
✗ Branch 7 → 8 not taken.
✗ Branch 7 → 204 not taken.
|
116 | HANDLE_UNRESOLVED_TYPE_PTR(iteratorOrIterableType) |
| 50 |
1/2✓ Branch 9 → 10 taken 116 times.
✗ Branch 9 → 205 not taken.
|
116 | iteratorOrIterableType = iteratorOrIterableType.removeReferenceWrapper(); |
| 51 | |||
| 52 | // Retrieve iterator type | ||
| 53 | 116 | QualType iteratorType = iteratorOrIterableType; | |
| 54 | |||
| 55 |
3/4✓ Branch 10 → 11 taken 116 times.
✗ Branch 10 → 346 not taken.
✓ Branch 11 → 12 taken 97 times.
✓ Branch 11 → 74 taken 19 times.
|
116 | if (iteratorOrIterableType.isIterable(node)) { |
| 56 | 97 | const QualType &iterableType = iteratorOrIterableType; | |
| 57 |
3/4✓ Branch 12 → 13 taken 97 times.
✗ Branch 12 → 346 not taken.
✓ Branch 13 → 14 taken 8 times.
✓ Branch 13 → 50 taken 89 times.
|
97 | if (iteratorOrIterableType.isArray()) { // Array |
| 58 |
2/4✓ Branch 16 → 17 taken 8 times.
✗ Branch 16 → 208 not taken.
✓ Branch 17 → 18 taken 8 times.
✗ Branch 17 → 206 not taken.
|
16 | const NameRegistryEntry *nameRegistryEntry = sourceFile->getNameRegistryEntry(ARRAY_ITERATOR_NAME); |
| 59 |
2/2✓ Branch 20 → 21 taken 1 time.
✓ Branch 20 → 29 taken 7 times.
|
8 | if (!nameRegistryEntry) { |
| 60 |
2/4✓ Branch 23 → 24 taken 1 time.
✗ Branch 23 → 214 not taken.
✓ Branch 24 → 25 taken 1 time.
✗ Branch 24 → 212 not taken.
|
1 | softError(node, UNKNOWN_DATATYPE, "Forgot to import \"std/iterator/array-iterator\"?"); |
| 61 |
1/2✓ Branch 27 → 28 taken 1 time.
✗ Branch 27 → 218 not taken.
|
1 | return nullptr; |
| 62 | } | ||
| 63 | 7 | nameRegistryEntry->targetEntry->used = nameRegistryEntry->importEntry->used = true; | |
| 64 | 7 | Scope *matchScope = nameRegistryEntry->targetScope->parent; | |
| 65 |
1/2✗ Branch 29 → 30 not taken.
✓ Branch 29 → 31 taken 7 times.
|
7 | assert(matchScope->type == ScopeType::GLOBAL); |
| 66 |
1/2✓ Branch 31 → 32 taken 7 times.
✗ Branch 31 → 236 not taken.
|
7 | QualType unsignedLongType(TY_LONG); |
| 67 |
1/2✓ Branch 32 → 33 taken 7 times.
✗ Branch 32 → 236 not taken.
|
7 | unsignedLongType.makeUnsigned(true); |
| 68 |
1/2✓ Branch 37 → 38 taken 7 times.
✗ Branch 37 → 219 not taken.
|
14 | const ArgList argTypes = {Arg(iterableType, false), Arg(unsignedLongType, false)}; |
| 69 |
1/2✓ Branch 39 → 40 taken 7 times.
✗ Branch 39 → 234 not taken.
|
7 | const QualType thisType(TY_DYN); |
| 70 |
2/4✓ Branch 43 → 44 taken 7 times.
✗ Branch 43 → 227 not taken.
✓ Branch 44 → 45 taken 7 times.
✗ Branch 44 → 225 not taken.
|
21 | node->getIteratorFct = FunctionManager::match(matchScope, "iterate", thisType, argTypes, {}, true, iteratorNode); |
| 71 | 7 | } else { // Struct, implementing Iterator interface | |
| 72 |
1/2✓ Branch 50 → 51 taken 89 times.
✗ Branch 50 → 346 not taken.
|
89 | Scope *matchScope = iterableType.getBodyScope(); |
| 73 |
2/4✓ Branch 55 → 56 taken 89 times.
✗ Branch 55 → 239 not taken.
✓ Branch 56 → 57 taken 89 times.
✗ Branch 56 → 237 not taken.
|
267 | node->getIteratorFct = FunctionManager::match(matchScope, "getIterator", iterableType, {}, {}, true, iteratorNode); |
| 74 | } | ||
| 75 |
2/2✓ Branch 62 → 63 taken 1 time.
✓ Branch 62 → 71 taken 95 times.
|
96 | if (node->getIteratorFct == nullptr) |
| 76 |
2/4✓ Branch 66 → 67 taken 1 time.
✗ Branch 66 → 252 not taken.
✓ Branch 67 → 68 taken 1 time.
✗ Branch 67 → 249 not taken.
|
3 | throw SemanticError(iteratorNode, INVALID_ITERATOR, "No getIterator() function found for the given iterable type"); |
| 77 | |||
| 78 | 95 | iteratorType = QualType(node->getIteratorFct->returnType); | |
| 79 | // Add anonymous symbol to keep track of dtor call, if non-trivially destructible | ||
| 80 |
2/4✓ Branch 71 → 72 taken 95 times.
✗ Branch 71 → 346 not taken.
✗ Branch 72 → 73 not taken.
✓ Branch 72 → 74 taken 95 times.
|
95 | if (!iteratorType.isTriviallyDestructible(iteratorNode)) |
| 81 | ✗ | currentScope->symbolTable.insertAnonymous(iteratorType, iteratorNode); | |
| 82 | } | ||
| 83 | |||
| 84 | // Change to foreach body scope | ||
| 85 |
2/4✓ Branch 74 → 75 taken 114 times.
✗ Branch 74 → 260 not taken.
✓ Branch 75 → 76 taken 114 times.
✗ Branch 75 → 258 not taken.
|
114 | ScopeHandle scopeHandle(this, node->getScopeId(), ScopeType::FOREACH_BODY); |
| 86 | |||
| 87 | // Check iterator type | ||
| 88 |
3/4✓ Branch 77 → 78 taken 114 times.
✗ Branch 77 → 344 not taken.
✓ Branch 78 → 79 taken 1 time.
✓ Branch 78 → 86 taken 113 times.
|
114 | if (!iteratorType.isIterator(node)) { |
| 89 | const std::string errMsg = | ||
| 90 | "Can only iterate over arrays or data structures, inheriting from IIterator or IIterable. You provided " + | ||
| 91 |
2/4✓ Branch 79 → 80 taken 1 time.
✗ Branch 79 → 264 not taken.
✓ Branch 80 → 81 taken 1 time.
✗ Branch 80 → 262 not taken.
|
1 | iteratorType.getName(false, true); |
| 92 |
1/2✓ Branch 82 → 83 taken 1 time.
✗ Branch 82 → 266 not taken.
|
1 | softError(node->iteratorAssign, OPERATOR_WRONG_DATA_TYPE, errMsg); |
| 93 |
1/2✓ Branch 83 → 84 taken 1 time.
✗ Branch 83 → 265 not taken.
|
1 | return nullptr; |
| 94 | 1 | } | |
| 95 |
1/2✓ Branch 86 → 87 taken 113 times.
✗ Branch 86 → 344 not taken.
|
113 | const QualTypeList &iteratorTemplateTypes = iteratorType.getTemplateTypes(); |
| 96 |
2/2✓ Branch 88 → 89 taken 1 time.
✓ Branch 88 → 99 taken 112 times.
|
113 | if (iteratorTemplateTypes.empty()) |
| 97 |
4/8✓ Branch 91 → 92 taken 1 time.
✗ Branch 91 → 271 not taken.
✓ Branch 92 → 93 taken 1 time.
✗ Branch 92 → 269 not taken.
✓ Branch 95 → 96 taken 1 time.
✗ Branch 95 → 275 not taken.
✓ Branch 96 → 97 taken 1 time.
✗ Branch 96 → 275 not taken.
|
3 | SOFT_ERROR_ER(node->iteratorAssign, INVALID_ITERATOR, |
| 98 | "Iterator has no generic arguments so that the item type could not be inferred") | ||
| 99 | |||
| 100 | 112 | const bool hasIdx = node->idxVarDecl; | |
| 101 |
2/2✓ Branch 99 → 100 taken 6 times.
✓ Branch 99 → 120 taken 106 times.
|
112 | if (hasIdx) { |
| 102 | // Visit index declaration or assignment | ||
| 103 |
2/4✓ Branch 100 → 101 taken 6 times.
✗ Branch 100 → 278 not taken.
✓ Branch 101 → 102 taken 6 times.
✗ Branch 101 → 276 not taken.
|
6 | auto indexType = std::any_cast<QualType>(visit(node->idxVarDecl)); |
| 104 |
2/6✓ Branch 103 → 104 taken 6 times.
✗ Branch 103 → 287 not taken.
✗ Branch 104 → 105 not taken.
✓ Branch 104 → 107 taken 6 times.
✗ Branch 105 → 106 not taken.
✗ Branch 105 → 279 not taken.
|
6 | HANDLE_UNRESOLVED_TYPE_PTR(indexType) |
| 105 | // Check if index type is int | ||
| 106 |
3/4✓ Branch 107 → 108 taken 6 times.
✗ Branch 107 → 287 not taken.
✓ Branch 108 → 109 taken 1 time.
✓ Branch 108 → 118 taken 5 times.
|
6 | if (!indexType.is(TY_LONG)) |
| 107 |
5/10✓ Branch 109 → 110 taken 1 time.
✗ Branch 109 → 284 not taken.
✓ Branch 110 → 111 taken 1 time.
✗ Branch 110 → 282 not taken.
✓ Branch 111 → 112 taken 1 time.
✗ Branch 111 → 280 not taken.
✓ Branch 114 → 115 taken 1 time.
✗ Branch 114 → 286 not taken.
✓ Branch 115 → 116 taken 1 time.
✗ Branch 115 → 286 not taken.
|
1 | SOFT_ERROR_ER(node->idxVarDecl, FOREACH_IDX_NOT_LONG, |
| 108 | "Index in foreach loop must be of type long. You provided " + indexType.getName(false)) | ||
| 109 | } | ||
| 110 | |||
| 111 | // Retrieve .get(), .getIdx(), .isValid() and .next() functions | ||
| 112 |
1/2✓ Branch 120 → 121 taken 111 times.
✗ Branch 120 → 344 not taken.
|
111 | Scope *matchScope = iteratorType.getBodyScope(); |
| 113 | 111 | QualType iteratorItemType; | |
| 114 |
2/2✓ Branch 121 → 122 taken 5 times.
✓ Branch 121 → 137 taken 106 times.
|
111 | if (hasIdx) { |
| 115 |
2/4✓ Branch 126 → 127 taken 5 times.
✗ Branch 126 → 290 not taken.
✓ Branch 127 → 128 taken 5 times.
✗ Branch 127 → 288 not taken.
|
15 | node->getIdxFct = FunctionManager::match(matchScope, "getIdx", iteratorType, {}, {}, false, node); |
| 116 |
1/2✗ Branch 132 → 133 not taken.
✓ Branch 132 → 134 taken 5 times.
|
5 | assert(node->getIdxFct != nullptr); |
| 117 |
1/2✓ Branch 134 → 135 taken 5 times.
✗ Branch 134 → 344 not taken.
|
5 | iteratorItemType = node->getIdxFct->returnType.getTemplateTypes().back(); |
| 118 | } else { | ||
| 119 |
2/4✓ Branch 141 → 142 taken 106 times.
✗ Branch 141 → 302 not taken.
✓ Branch 142 → 143 taken 106 times.
✗ Branch 142 → 300 not taken.
|
318 | node->getFct = FunctionManager::match(matchScope, "get", iteratorType, {}, {}, false, node); |
| 120 |
1/2✗ Branch 147 → 148 not taken.
✓ Branch 147 → 149 taken 106 times.
|
106 | assert(node->getFct != nullptr); |
| 121 | 106 | iteratorItemType = node->getFct->returnType; | |
| 122 | } | ||
| 123 |
2/4✓ Branch 154 → 155 taken 111 times.
✗ Branch 154 → 314 not taken.
✓ Branch 155 → 156 taken 111 times.
✗ Branch 155 → 312 not taken.
|
333 | node->isValidFct = FunctionManager::match(matchScope, "isValid", iteratorType, {}, {}, false, node); |
| 124 |
1/2✗ Branch 160 → 161 not taken.
✓ Branch 160 → 162 taken 111 times.
|
111 | assert(node->isValidFct != nullptr); |
| 125 |
2/4✓ Branch 166 → 167 taken 111 times.
✗ Branch 166 → 326 not taken.
✓ Branch 167 → 168 taken 111 times.
✗ Branch 167 → 324 not taken.
|
333 | node->nextFct = FunctionManager::match(matchScope, "next", iteratorType, {}, {}, false, node); |
| 126 |
1/2✗ Branch 172 → 173 not taken.
✓ Branch 172 → 174 taken 111 times.
|
111 | assert(node->nextFct != nullptr); |
| 127 | |||
| 128 | // Retrieve item variable entry | ||
| 129 |
1/2✓ Branch 174 → 175 taken 111 times.
✗ Branch 174 → 344 not taken.
|
111 | SymbolTableEntry *itemVarSymbol = currentScope->lookupStrict(node->itemVarDecl->varName); |
| 130 |
1/2✗ Branch 177 → 178 not taken.
✓ Branch 177 → 179 taken 111 times.
|
111 | assert(itemVarSymbol != nullptr); |
| 131 | |||
| 132 | // Check type of the item | ||
| 133 |
2/4✓ Branch 179 → 180 taken 111 times.
✗ Branch 179 → 338 not taken.
✓ Branch 180 → 181 taken 111 times.
✗ Branch 180 → 336 not taken.
|
111 | auto itemType = std::any_cast<QualType>(visit(node->itemVarDecl)); |
| 134 |
2/6✓ Branch 182 → 183 taken 111 times.
✗ Branch 182 → 344 not taken.
✗ Branch 183 → 184 not taken.
✓ Branch 183 → 186 taken 111 times.
✗ Branch 184 → 185 not taken.
✗ Branch 184 → 339 not taken.
|
111 | HANDLE_UNRESOLVED_TYPE_PTR(itemType) |
| 135 |
3/4✓ Branch 186 → 187 taken 111 times.
✗ Branch 186 → 344 not taken.
✓ Branch 187 → 188 taken 4 times.
✓ Branch 187 → 190 taken 107 times.
|
111 | if (itemType.is(TY_DYN)) { // Perform type inference |
| 136 | // Update evaluated symbol type of the declaration data type | ||
| 137 |
1/2✓ Branch 188 → 189 taken 4 times.
✗ Branch 188 → 344 not taken.
|
4 | node->itemVarDecl->dataType->setEvaluatedSymbolType(iteratorItemType, manIdx); |
| 138 | // Update item type | ||
| 139 | 4 | itemType = iteratorItemType; | |
| 140 | } else { | ||
| 141 | // Check item type | ||
| 142 | 107 | const ExprResult itemResult = {itemType, itemVarSymbol}; | |
| 143 | 107 | const ExprResult iteratorItemResult = {iteratorItemType, nullptr /* always a temporary */}; | |
| 144 |
2/2✓ Branch 190 → 191 taken 106 times.
✓ Branch 190 → 340 taken 1 time.
|
107 | (void)opRuleManager.getAssignResultType(node->itemVarDecl, itemResult, iteratorItemResult, true, false, ERROR_FOREACH_ITEM); |
| 145 | } | ||
| 146 | |||
| 147 | // Update type of item | ||
| 148 |
1/2✓ Branch 192 → 193 taken 110 times.
✗ Branch 192 → 344 not taken.
|
110 | itemVarSymbol->updateType(itemType, true); |
| 149 | |||
| 150 | // Visit body | ||
| 151 |
1/2✓ Branch 193 → 194 taken 110 times.
✗ Branch 193 → 342 not taken.
|
110 | visit(node->body); |
| 152 | |||
| 153 |
1/2✓ Branch 195 → 196 taken 110 times.
✗ Branch 195 → 343 not taken.
|
110 | return nullptr; |
| 154 | 114 | } | |
| 155 | |||
| 156 | 733 | std::any TypeChecker::visitWhileLoop(WhileLoopNode *node) { | |
| 157 | // Change to while body scope | ||
| 158 |
2/4✓ Branch 2 → 3 taken 733 times.
✗ Branch 2 → 33 not taken.
✓ Branch 3 → 4 taken 733 times.
✗ Branch 3 → 31 not taken.
|
733 | ScopeHandle scopeHandle(this, node->getScopeId(), ScopeType::WHILE_BODY); |
| 159 | |||
| 160 | // Visit condition | ||
| 161 |
3/4✓ Branch 5 → 6 taken 732 times.
✓ Branch 5 → 37 taken 1 time.
✓ Branch 6 → 7 taken 732 times.
✗ Branch 6 → 35 not taken.
|
733 | const QualType conditionType = std::any_cast<ExprResult>(visit(node->condition)).type; |
| 162 |
2/6✓ Branch 8 → 9 taken 732 times.
✗ Branch 8 → 49 not taken.
✗ Branch 9 → 10 not taken.
✓ Branch 9 → 12 taken 732 times.
✗ Branch 10 → 11 not taken.
✗ Branch 10 → 39 not taken.
|
732 | HANDLE_UNRESOLVED_TYPE_PTR(conditionType) |
| 163 | // Check if condition evaluates to bool | ||
| 164 |
3/4✓ Branch 12 → 13 taken 732 times.
✗ Branch 12 → 49 not taken.
✓ Branch 13 → 14 taken 1 time.
✓ Branch 13 → 24 taken 731 times.
|
732 | if (!conditionType.is(TY_BOOL)) |
| 165 |
4/8✓ Branch 16 → 17 taken 1 time.
✗ Branch 16 → 42 not taken.
✓ Branch 17 → 18 taken 1 time.
✗ Branch 17 → 40 not taken.
✓ Branch 20 → 21 taken 1 time.
✗ Branch 20 → 46 not taken.
✓ Branch 21 → 22 taken 1 time.
✗ Branch 21 → 46 not taken.
|
3 | SOFT_ERROR_ER(node->condition, CONDITION_MUST_BE_BOOL, "While loop condition must be of type bool") |
| 166 | |||
| 167 | // Visit body | ||
| 168 |
1/2✓ Branch 24 → 25 taken 731 times.
✗ Branch 24 → 47 not taken.
|
731 | visit(node->body); |
| 169 | |||
| 170 |
1/2✓ Branch 26 → 27 taken 731 times.
✗ Branch 26 → 48 not taken.
|
731 | return nullptr; |
| 171 | 733 | } | |
| 172 | |||
| 173 | 9 | std::any TypeChecker::visitDoWhileLoop(DoWhileLoopNode *node) { | |
| 174 | // Change to while body scope | ||
| 175 |
2/4✓ Branch 2 → 3 taken 9 times.
✗ Branch 2 → 33 not taken.
✓ Branch 3 → 4 taken 9 times.
✗ Branch 3 → 31 not taken.
|
9 | ScopeHandle scopeHandle(this, node->getScopeId(), ScopeType::WHILE_BODY); |
| 176 | |||
| 177 | // Visit body | ||
| 178 |
1/2✓ Branch 5 → 6 taken 9 times.
✗ Branch 5 → 35 not taken.
|
9 | visit(node->body); |
| 179 | |||
| 180 | // Visit condition | ||
| 181 |
2/4✓ Branch 7 → 8 taken 9 times.
✗ Branch 7 → 38 not taken.
✓ Branch 8 → 9 taken 9 times.
✗ Branch 8 → 36 not taken.
|
9 | const QualType conditionType = std::any_cast<ExprResult>(visit(node->condition)).type; |
| 182 |
2/6✓ Branch 10 → 11 taken 9 times.
✗ Branch 10 → 49 not taken.
✗ Branch 11 → 12 not taken.
✓ Branch 11 → 14 taken 9 times.
✗ Branch 12 → 13 not taken.
✗ Branch 12 → 40 not taken.
|
9 | HANDLE_UNRESOLVED_TYPE_PTR(conditionType) |
| 183 | // Check if condition evaluates to bool | ||
| 184 |
3/4✓ Branch 14 → 15 taken 9 times.
✗ Branch 14 → 49 not taken.
✓ Branch 15 → 16 taken 1 time.
✓ Branch 15 → 26 taken 8 times.
|
9 | if (!conditionType.is(TY_BOOL)) |
| 185 |
4/8✓ Branch 18 → 19 taken 1 time.
✗ Branch 18 → 43 not taken.
✓ Branch 19 → 20 taken 1 time.
✗ Branch 19 → 41 not taken.
✓ Branch 22 → 23 taken 1 time.
✗ Branch 22 → 47 not taken.
✓ Branch 23 → 24 taken 1 time.
✗ Branch 23 → 47 not taken.
|
3 | SOFT_ERROR_ER(node->condition, CONDITION_MUST_BE_BOOL, "Do-While loop condition must be of type bool") |
| 186 | |||
| 187 |
1/2✓ Branch 26 → 27 taken 8 times.
✗ Branch 26 → 48 not taken.
|
8 | return nullptr; |
| 188 | 9 | } | |
| 189 | |||
| 190 | 4269 | std::any TypeChecker::visitIfStmt(IfStmtNode *node) { | |
| 191 | // Change to then body scope | ||
| 192 |
2/4✓ Branch 2 → 3 taken 4269 times.
✗ Branch 2 → 41 not taken.
✓ Branch 3 → 4 taken 4269 times.
✗ Branch 3 → 39 not taken.
|
4269 | ScopeHandle scopeHandle(this, node->getScopeId(), ScopeType::IF_ELSE_BODY); |
| 193 | |||
| 194 | // Visit condition | ||
| 195 |
3/4✓ Branch 5 → 6 taken 4267 times.
✓ Branch 5 → 45 taken 2 times.
✓ Branch 6 → 7 taken 4267 times.
✗ Branch 6 → 43 not taken.
|
4269 | const QualType conditionType = std::any_cast<ExprResult>(visit(node->condition)).type; |
| 196 |
4/6✓ Branch 8 → 9 taken 4267 times.
✗ Branch 8 → 59 not taken.
✓ Branch 9 → 10 taken 1 time.
✓ Branch 9 → 12 taken 4266 times.
✓ Branch 10 → 11 taken 1 time.
✗ Branch 10 → 47 not taken.
|
4267 | HANDLE_UNRESOLVED_TYPE_PTR(conditionType) |
| 197 | // Check if condition evaluates to bool | ||
| 198 |
3/4✓ Branch 12 → 13 taken 4266 times.
✗ Branch 12 → 59 not taken.
✓ Branch 13 → 14 taken 2 times.
✓ Branch 13 → 24 taken 4264 times.
|
4266 | if (!conditionType.is(TY_BOOL)) |
| 199 |
4/8✓ Branch 16 → 17 taken 2 times.
✗ Branch 16 → 50 not taken.
✓ Branch 17 → 18 taken 2 times.
✗ Branch 17 → 48 not taken.
✓ Branch 20 → 21 taken 2 times.
✗ Branch 20 → 54 not taken.
✓ Branch 21 → 22 taken 2 times.
✗ Branch 21 → 54 not taken.
|
6 | SOFT_ERROR_ER(node->condition, CONDITION_MUST_BE_BOOL, "If condition must be of type bool") |
| 200 | |||
| 201 | // Warning for bool assignment | ||
| 202 |
2/2✓ Branch 24 → 25 taken 1 time.
✓ Branch 24 → 27 taken 4263 times.
|
4264 | if (node->condition->op == AssignExprNode::AssignOp::OP_ASSIGN) |
| 203 |
1/2✓ Branch 25 → 26 taken 1 time.
✗ Branch 25 → 55 not taken.
|
1 | sourceFile->compilerOutput.warnings.emplace_back(node->condition->codeLoc, BOOL_ASSIGN_AS_CONDITION, |
| 204 | "If you want to compare the values, use '=='"); | ||
| 205 | |||
| 206 | // Visit body | ||
| 207 |
1/2✓ Branch 27 → 28 taken 4264 times.
✗ Branch 27 → 56 not taken.
|
4264 | visit(node->thenBody); |
| 208 | |||
| 209 | // Leave then body scope | ||
| 210 |
1/2✓ Branch 29 → 30 taken 4264 times.
✗ Branch 29 → 59 not taken.
|
4264 | scopeHandle.leaveScopeEarly(); |
| 211 | |||
| 212 | // Visit else statement if existing | ||
| 213 |
2/2✓ Branch 30 → 31 taken 186 times.
✓ Branch 30 → 34 taken 4078 times.
|
4264 | if (node->elseStmt) |
| 214 |
1/2✓ Branch 31 → 32 taken 186 times.
✗ Branch 31 → 57 not taken.
|
186 | visit(node->elseStmt); |
| 215 | |||
| 216 |
1/2✓ Branch 34 → 35 taken 4264 times.
✗ Branch 34 → 58 not taken.
|
4264 | return nullptr; |
| 217 | 4269 | } | |
| 218 | |||
| 219 | 186 | std::any TypeChecker::visitElseStmt(ElseStmtNode *node) { | |
| 220 | // Visit if statement in the case of an else if branch | ||
| 221 |
2/2✓ Branch 2 → 3 taken 53 times.
✓ Branch 2 → 7 taken 133 times.
|
186 | if (node->isElseIf) { |
| 222 |
1/2✓ Branch 3 → 4 taken 53 times.
✗ Branch 3 → 16 not taken.
|
53 | visit(node->ifStmt); |
| 223 |
1/2✓ Branch 5 → 6 taken 53 times.
✗ Branch 5 → 17 not taken.
|
53 | return nullptr; |
| 224 | } | ||
| 225 | |||
| 226 | // Change to else body scope | ||
| 227 |
2/4✓ Branch 7 → 8 taken 133 times.
✗ Branch 7 → 20 not taken.
✓ Branch 8 → 9 taken 133 times.
✗ Branch 8 → 18 not taken.
|
133 | ScopeHandle scopeHandle(this, node->getScopeId(), ScopeType::IF_ELSE_BODY); |
| 228 | |||
| 229 | // Visit body | ||
| 230 |
1/2✓ Branch 10 → 11 taken 133 times.
✗ Branch 10 → 22 not taken.
|
133 | visit(node->body); |
| 231 | |||
| 232 |
1/2✓ Branch 12 → 13 taken 133 times.
✗ Branch 12 → 23 not taken.
|
133 | return nullptr; |
| 233 | 133 | } | |
| 234 | |||
| 235 | 12 | std::any TypeChecker::visitSwitchStmt(SwitchStmtNode *node) { | |
| 236 | // Check expression type | ||
| 237 |
2/4✓ Branch 2 → 3 taken 12 times.
✗ Branch 2 → 56 not taken.
✓ Branch 3 → 4 taken 12 times.
✗ Branch 3 → 54 not taken.
|
12 | const QualType exprType = std::any_cast<ExprResult>(visit(node->assignExpr)).type; |
| 238 |
2/6✓ Branch 5 → 6 taken 12 times.
✗ Branch 5 → 83 not taken.
✗ Branch 6 → 7 not taken.
✓ Branch 6 → 9 taken 12 times.
✗ Branch 7 → 8 not taken.
✗ Branch 7 → 58 not taken.
|
12 | HANDLE_UNRESOLVED_TYPE_PTR(exprType) |
| 239 |
3/4✓ Branch 9 → 10 taken 12 times.
✗ Branch 9 → 59 not taken.
✓ Branch 10 → 11 taken 1 time.
✓ Branch 10 → 21 taken 11 times.
|
12 | if (!exprType.isOneOf({TY_INT, TY_SHORT, TY_LONG, TY_BYTE, TY_CHAR, TY_BOOL})) |
| 240 |
4/8✓ Branch 13 → 14 taken 1 time.
✗ Branch 13 → 62 not taken.
✓ Branch 14 → 15 taken 1 time.
✗ Branch 14 → 60 not taken.
✓ Branch 17 → 18 taken 1 time.
✗ Branch 17 → 66 not taken.
✓ Branch 18 → 19 taken 1 time.
✗ Branch 18 → 66 not taken.
|
3 | SOFT_ERROR_ER(node->assignExpr, SWITCH_EXPR_MUST_BE_PRIMITIVE, |
| 241 | "Switch expression must be of int, short, long, byte, char or bool type") | ||
| 242 | |||
| 243 | // Visit children | ||
| 244 |
1/2✓ Branch 21 → 22 taken 11 times.
✗ Branch 21 → 67 not taken.
|
11 | visitChildren(node); |
| 245 | |||
| 246 | // Check if case constant types match switch expression type | ||
| 247 |
2/2✓ Branch 49 → 25 taken 53 times.
✓ Branch 49 → 50 taken 9 times.
|
62 | for (const CaseBranchNode *caseBranchNode : node->caseBranches) |
| 248 |
2/2✓ Branch 46 → 28 taken 70 times.
✓ Branch 46 → 47 taken 51 times.
|
121 | for (CaseConstantNode *constantNode : caseBranchNode->caseConstants) { |
| 249 |
2/4✓ Branch 29 → 30 taken 70 times.
✗ Branch 29 → 70 not taken.
✓ Branch 30 → 31 taken 70 times.
✗ Branch 30 → 68 not taken.
|
70 | const QualType constantType = std::any_cast<ExprResult>(visit(constantNode)).type; |
| 250 |
3/4✓ Branch 32 → 33 taken 70 times.
✗ Branch 32 → 79 not taken.
✓ Branch 33 → 34 taken 2 times.
✓ Branch 33 → 44 taken 68 times.
|
70 | if (!constantType.matches(exprType, false, true, true)) |
| 251 |
4/8✓ Branch 36 → 37 taken 2 times.
✗ Branch 36 → 74 not taken.
✓ Branch 37 → 38 taken 2 times.
✗ Branch 37 → 72 not taken.
✓ Branch 40 → 41 taken 2 times.
✗ Branch 40 → 78 not taken.
✓ Branch 41 → 42 taken 2 times.
✗ Branch 41 → 78 not taken.
|
6 | SOFT_ERROR_ER(constantNode, SWITCH_CASE_TYPE_MISMATCH, "Case value type does not match the switch expression type") |
| 252 | } | ||
| 253 | |||
| 254 |
1/2✓ Branch 50 → 51 taken 9 times.
✗ Branch 50 → 82 not taken.
|
9 | return nullptr; |
| 255 | } | ||
| 256 | |||
| 257 | 53 | std::any TypeChecker::visitCaseBranch(CaseBranchNode *node) { | |
| 258 | // Change to case body scope | ||
| 259 |
2/4✓ Branch 2 → 3 taken 53 times.
✗ Branch 2 → 21 not taken.
✓ Branch 3 → 4 taken 53 times.
✗ Branch 3 → 19 not taken.
|
53 | ScopeHandle scopeHandle(this, node->getScopeId(), ScopeType::CASE_BODY); |
| 260 | |||
| 261 | // Visit constant list | ||
| 262 |
2/2✓ Branch 12 → 7 taken 70 times.
✓ Branch 12 → 13 taken 53 times.
|
123 | for (CaseConstantNode *constant : node->caseConstants) |
| 263 |
1/2✓ Branch 8 → 9 taken 70 times.
✗ Branch 8 → 23 not taken.
|
70 | visit(constant); |
| 264 | |||
| 265 | // Visit body | ||
| 266 |
1/2✓ Branch 13 → 14 taken 53 times.
✗ Branch 13 → 25 not taken.
|
53 | visit(node->body); |
| 267 | |||
| 268 |
1/2✓ Branch 15 → 16 taken 53 times.
✗ Branch 15 → 26 not taken.
|
106 | return nullptr; |
| 269 | 53 | } | |
| 270 | |||
| 271 | 140 | std::any TypeChecker::visitCaseConstant(CaseConstantNode *node) { | |
| 272 | // If we have a normal constant, we can take the symbol type from there | ||
| 273 |
2/2✓ Branch 2 → 3 taken 36 times.
✓ Branch 2 → 4 taken 104 times.
|
140 | if (node->constant) |
| 274 | 36 | return visit(node->constant); | |
| 275 | |||
| 276 | // Check if a local or global variable can be found by searching for the name | ||
| 277 |
2/2✓ Branch 5 → 6 taken 30 times.
✓ Branch 5 → 11 taken 74 times.
|
104 | if (node->identifierFragments.size() == 1) |
| 278 | 60 | node->entry = currentScope->lookup(node->identifierFragments.back()); | |
| 279 | |||
| 280 | // If no local or global was found, search in the name registry | ||
| 281 |
2/2✓ Branch 11 → 12 taken 37 times.
✓ Branch 11 → 24 taken 67 times.
|
104 | if (!node->entry) { |
| 282 | 37 | const NameRegistryEntry *registryEntry = sourceFile->getNameRegistryEntry(node->fqIdentifier); | |
| 283 |
1/2✗ Branch 13 → 14 not taken.
✓ Branch 13 → 23 taken 37 times.
|
37 | if (!registryEntry) |
| 284 | ✗ | SOFT_ERROR_ER(node, REFERENCED_UNDEFINED_VARIABLE, "The variable '" + node->fqIdentifier + "' could not be found") | |
| 285 | 37 | node->entry = registryEntry->targetEntry; | |
| 286 | } | ||
| 287 |
1/2✗ Branch 24 → 25 not taken.
✓ Branch 24 → 26 taken 104 times.
|
104 | assert(node->entry != nullptr); |
| 288 | |||
| 289 | // Check for the correct type | ||
| 290 | 104 | const QualType &qualType = node->entry->getQualType(); | |
| 291 |
3/4✓ Branch 27 → 28 taken 104 times.
✗ Branch 27 → 51 not taken.
✓ Branch 28 → 29 taken 2 times.
✓ Branch 28 → 39 taken 102 times.
|
104 | if (!qualType.isOneOf({TY_INT, TY_SHORT, TY_LONG, TY_BYTE, TY_CHAR, TY_BOOL})) |
| 292 |
4/8✓ Branch 31 → 32 taken 2 times.
✗ Branch 31 → 54 not taken.
✓ Branch 32 → 33 taken 2 times.
✗ Branch 32 → 52 not taken.
✓ Branch 35 → 36 taken 2 times.
✗ Branch 35 → 58 not taken.
✓ Branch 36 → 37 taken 2 times.
✗ Branch 36 → 58 not taken.
|
6 | SOFT_ERROR_ER(node, CASE_CONSTANT_INVALID_TYPE, "Case constants must be of type int, short, long, byte, char, bool or enum") |
| 293 | |||
| 294 |
2/4✓ Branch 39 → 40 taken 102 times.
✗ Branch 39 → 59 not taken.
✓ Branch 40 → 41 taken 102 times.
✗ Branch 40 → 59 not taken.
|
102 | return ExprResult{node->setEvaluatedSymbolType(qualType, manIdx)}; |
| 295 | } | ||
| 296 | |||
| 297 | 6 | std::any TypeChecker::visitDefaultBranch(DefaultBranchNode *node) { | |
| 298 | // Change to default body scope | ||
| 299 |
2/4✓ Branch 2 → 3 taken 6 times.
✗ Branch 2 → 13 not taken.
✓ Branch 3 → 4 taken 6 times.
✗ Branch 3 → 11 not taken.
|
6 | ScopeHandle scopeHandle(this, node->getScopeId(), ScopeType::DEFAULT_BODY); |
| 300 | |||
| 301 | // Visit body | ||
| 302 |
1/2✓ Branch 5 → 6 taken 6 times.
✗ Branch 5 → 15 not taken.
|
6 | visit(node->body); |
| 303 | |||
| 304 |
1/2✓ Branch 7 → 8 taken 6 times.
✗ Branch 7 → 16 not taken.
|
12 | return nullptr; |
| 305 | 6 | } | |
| 306 | |||
| 307 | 30 | std::any TypeChecker::visitAnonymousBlockStmt(AnonymousBlockStmtNode *node) { | |
| 308 | // Change to anonymous scope body scope | ||
| 309 |
2/4✓ Branch 2 → 3 taken 30 times.
✗ Branch 2 → 13 not taken.
✓ Branch 3 → 4 taken 30 times.
✗ Branch 3 → 11 not taken.
|
30 | ScopeHandle scopeHandle(this, node->getScopeId(), ScopeType::ANONYMOUS_BLOCK_BODY); |
| 310 | |||
| 311 | // Visit body | ||
| 312 |
1/2✓ Branch 5 → 6 taken 30 times.
✗ Branch 5 → 15 not taken.
|
30 | visit(node->body); |
| 313 | |||
| 314 |
1/2✓ Branch 7 → 8 taken 30 times.
✗ Branch 7 → 16 not taken.
|
60 | return nullptr; |
| 315 | 30 | } | |
| 316 | |||
| 317 | } // namespace spice::compiler | ||
| 318 |