GCC Code Coverage Report


Directory: ../
Coverage: low: ≥ 0% medium: ≥ 75.0% high: ≥ 90.0%
Coverage Exec / Excl / Total
Lines: 98.4% 122 / 0 / 124
Functions: 100.0% 15 / 0 / 15
Branches: 64.7% 145 / 0 / 224

src/symboltablebuilder/SymbolTable.cpp
Line Branch Exec Source
1 // Copyright (c) 2021-2026 ChilliBits. All rights reserved.
2
3 #include "SymbolTable.h"
4
5 #include <SourceFile.h>
6 #include <ast/ASTNodes.h>
7 #include <symboltablebuilder/Scope.h>
8 #include <symboltablebuilder/SymbolTableBuilder.h>
9 #include <util/CodeLoc.h>
10 #include <util/CompilerWarning.h>
11
12 namespace spice::compiler {
13
14 /**
15 * Insert a new symbol into the current symbol table. If it is a parameter, append its name to the paramNames vector
16 *
17 * @param name Name of the symbol
18 * @param declNode AST node where the symbol is declared
19 * @param isAnonymousSymbol If this symbol should be anonymous
20 * @return Inserted entry
21 */
22 145932 SymbolTableEntry *SymbolTable::insert(const std::string &name, ASTNode *declNode, bool isAnonymousSymbol) {
23 145932 const bool isGlobal = parent == nullptr;
24 145932 size_t orderIndex = SIZE_MAX;
25
2/2
✓ Branch 2 → 3 taken 137983 times.
✓ Branch 2 → 5 taken 7949 times.
145932 if (!isAnonymousSymbol)
26 1281153 orderIndex = std::ranges::count_if(symbols, [](const auto &entry) { return !entry.second.anonymous; });
27 // Insert into symbols map. The type is 'dyn', because concrete types are determined by the type checker later on
28
1/2
✗ Branch 6 → 7 not taken.
✓ Branch 6 → 8 taken 145932 times.
145932 assert(!symbols.contains(name));
29
4/8
✓ Branch 8 → 9 taken 145932 times.
✗ Branch 8 → 41 not taken.
✓ Branch 9 → 10 taken 145932 times.
✗ Branch 9 → 40 not taken.
✓ Branch 10 → 11 taken 145932 times.
✗ Branch 10 → 38 not taken.
✓ Branch 11 → 12 taken 145932 times.
✗ Branch 11 → 36 not taken.
145932 symbols.emplace(name, SymbolTableEntry(name, QualType(TY_INVALID), scope, declNode, orderIndex, isGlobal));
30 // Set entry to declared
31 145932 SymbolTableEntry *entry = &symbols.at(name);
32
1/2
✓ Branch 15 → 16 taken 145932 times.
✗ Branch 15 → 43 not taken.
145932 entry->updateState(DECLARED, declNode);
33
34 // Check if shadowed
35
10/10
✓ Branch 16 → 17 taken 121235 times.
✓ Branch 16 → 24 taken 24697 times.
✓ Branch 18 → 19 taken 1910 times.
✓ Branch 18 → 24 taken 119325 times.
✓ Branch 20 → 21 taken 18 times.
✓ Branch 20 → 24 taken 1892 times.
✓ Branch 22 → 23 taken 2 times.
✓ Branch 22 → 24 taken 16 times.
✓ Branch 25 → 26 taken 2 times.
✓ Branch 25 → 34 taken 145930 times.
145932 if (parent != nullptr && parent->lookup(name) != nullptr && !declNode->isParam() && name != RETURN_VARIABLE_NAME) {
36
2/4
✓ Branch 26 → 27 taken 2 times.
✗ Branch 26 → 46 not taken.
✓ Branch 27 → 28 taken 2 times.
✗ Branch 27 → 44 not taken.
2 const std::string warningMsg = "Variable '" + name + "' shadows a variable in a parent scope";
37
1/2
✓ Branch 29 → 30 taken 2 times.
✗ Branch 29 → 49 not taken.
2 const CompilerWarning warning(declNode->codeLoc, SHADOWED_VARIABLE, warningMsg);
38
1/2
✓ Branch 30 → 31 taken 2 times.
✗ Branch 30 → 47 not taken.
2 scope->sourceFile->compilerOutput.warnings.push_back(warning);
39 2 }
40
41 145932 return entry;
42 }
43
44 /**
45 * Insert a new anonymous symbol into the current symbol table.
46 * The anonymous symbol will be identified via the definition code location
47 *
48 * @param qualType Type of the symbol
49 * @param declNode AST node where the anonymous symbol is declared
50 * @param numericSuffix Custom numeric suffix
51 * @return Inserted entry
52 */
53 7975 SymbolTableEntry *SymbolTable::insertAnonymous(const QualType &qualType, ASTNode *declNode, size_t numericSuffix) {
54 // Check if the anonymous entry already exists
55
3/4
✓ Branch 2 → 3 taken 7975 times.
✗ Branch 2 → 34 not taken.
✓ Branch 3 → 4 taken 26 times.
✓ Branch 3 → 5 taken 7949 times.
7975 if (SymbolTableEntry *anonSymbol = lookupAnonymous(declNode, numericSuffix))
56 26 return anonSymbol;
57 // Otherwise, create an anonymous entry
58
1/2
✓ Branch 5 → 6 taken 7949 times.
✗ Branch 5 → 34 not taken.
7949 std::stringstream name;
59
5/10
✓ Branch 6 → 7 taken 7949 times.
✗ Branch 6 → 32 not taken.
✓ Branch 7 → 8 taken 7949 times.
✗ Branch 7 → 26 not taken.
✓ Branch 8 → 9 taken 7949 times.
✗ Branch 8 → 24 not taken.
✓ Branch 9 → 10 taken 7949 times.
✗ Branch 9 → 24 not taken.
✓ Branch 10 → 11 taken 7949 times.
✗ Branch 10 → 24 not taken.
7949 name << "anon." << declNode->codeLoc.toString() << "." << reinterpret_cast<size_t>(declNode);
60
2/2
✓ Branch 12 → 13 taken 265 times.
✓ Branch 12 → 15 taken 7684 times.
7949 if (numericSuffix > 0)
61
2/4
✓ Branch 13 → 14 taken 265 times.
✗ Branch 13 → 32 not taken.
✓ Branch 14 → 15 taken 265 times.
✗ Branch 14 → 32 not taken.
265 name << "." << numericSuffix;
62
2/4
✓ Branch 15 → 16 taken 7949 times.
✗ Branch 15 → 29 not taken.
✓ Branch 16 → 17 taken 7949 times.
✗ Branch 16 → 27 not taken.
7949 SymbolTableEntry *anonSymbol = insert(name.str(), declNode, true);
63
1/2
✓ Branch 18 → 19 taken 7949 times.
✗ Branch 18 → 32 not taken.
7949 anonSymbol->updateType(qualType, false);
64
1/2
✓ Branch 19 → 20 taken 7949 times.
✗ Branch 19 → 30 not taken.
7949 anonSymbol->updateState(DECLARED, declNode);
65
1/2
✓ Branch 20 → 21 taken 7949 times.
✗ Branch 20 → 31 not taken.
7949 anonSymbol->updateState(INITIALIZED, declNode);
66 7949 anonSymbol->anonymous = true;
67 7949 anonSymbol->used = true;
68 7949 return anonSymbol;
69 7949 }
70
71 /**
72 * Copy a symbol by its name
73 *
74 * @param originalName Original symbol name
75 * @param newName New symbol name
76 * @return Copied entry
77 */
78 13236 SymbolTableEntry *SymbolTable::copySymbol(const std::string &originalName, const std::string &newName) {
79
1/2
✓ Branch 2 → 3 taken 13236 times.
✗ Branch 2 → 13 not taken.
13236 SymbolTableEntry *entryToCopy = lookupStrict(originalName);
80
1/2
✗ Branch 3 → 4 not taken.
✓ Branch 3 → 5 taken 13236 times.
13236 assert(entryToCopy != nullptr);
81
1/2
✓ Branch 5 → 6 taken 13236 times.
✗ Branch 5 → 13 not taken.
13236 auto [it, success] = symbols.emplace(newName, *entryToCopy);
82
1/2
✗ Branch 8 → 9 not taken.
✓ Branch 8 → 10 taken 13236 times.
13236 assert(success);
83 13236 return &it->second;
84 }
85
86 /**
87 * Check if a symbol exists in the current or any parent scope and return it if possible
88 *
89 * @param name Name of the desired symbol
90 * @return Desired symbol / nullptr if the symbol was not found
91 */
92 768135 SymbolTableEntry *SymbolTable::lookup(const std::string &name) { // NOLINT(misc-no-recursion)
93 // Check if the symbol exists in the current scope. If yes, take it
94
2/2
✓ Branch 3 → 4 taken 244422 times.
✓ Branch 3 → 5 taken 523713 times.
768135 if (SymbolTableEntry *entry = lookupStrict(name))
95 244422 return entry;
96
97 // Symbol was not found in the current scope
98 // We reached the root scope, the symbol does not exist at all
99
2/2
✓ Branch 5 → 6 taken 210985 times.
✓ Branch 5 → 7 taken 312728 times.
523713 if (!parent)
100 210985 return nullptr;
101 // If we search for the result variable, we want to stop the search when exiting a lambda body
102
6/6
✓ Branch 8 → 9 taken 18249 times.
✓ Branch 8 → 11 taken 294479 times.
✓ Branch 9 → 10 taken 3 times.
✓ Branch 9 → 11 taken 18246 times.
✓ Branch 12 → 13 taken 3 times.
✓ Branch 12 → 14 taken 312725 times.
312728 if (name == RETURN_VARIABLE_NAME && scope->type == ScopeType::LAMBDA_BODY)
103 3 return nullptr;
104 // If there is a parent scope, continue the search there
105 312725 SymbolTableEntry *entry = parent->lookup(name);
106 // Symbol was also not found in all the parent scopes, return nullptr
107
2/2
✓ Branch 15 → 16 taken 164400 times.
✓ Branch 15 → 17 taken 148325 times.
312725 if (!entry)
108 164400 return nullptr;
109 // Check if this scope requires capturing and capture the variable if appropriate
110
9/14
✓ Branch 17 → 18 taken 55 times.
✓ Branch 17 → 24 taken 148270 times.
✓ Branch 18 → 19 taken 55 times.
✗ Branch 18 → 37 not taken.
✓ Branch 19 → 20 taken 55 times.
✗ Branch 19 → 24 not taken.
✓ Branch 20 → 21 taken 55 times.
✗ Branch 20 → 37 not taken.
✓ Branch 21 → 22 taken 55 times.
✗ Branch 21 → 37 not taken.
✓ Branch 22 → 23 taken 55 times.
✗ Branch 22 → 24 not taken.
✓ Branch 25 → 26 taken 55 times.
✓ Branch 25 → 35 taken 148270 times.
148325 if (capturingRequired && !captures.contains(name) && !entry->getQualType().isOneOf({TY_IMPORT, TY_FUNCTION, TY_PROCEDURE})) {
111 // We need to make the symbol volatile if we are in an async scope and try to access a symbol that is not in an async scope
112
3/4
✓ Branch 27 → 28 taken 8 times.
✓ Branch 27 → 31 taken 47 times.
✓ Branch 29 → 30 taken 8 times.
✗ Branch 29 → 31 not taken.
55 entry->isVolatile = scope->isInAsyncScope() && !entry->scope->isInAsyncScope();
113 // Add the capture to the current scope
114
2/4
✓ Branch 32 → 33 taken 55 times.
✗ Branch 32 → 38 not taken.
✓ Branch 33 → 34 taken 55 times.
✗ Branch 33 → 38 not taken.
55 captures.emplace(name, Capture(entry));
115 }
116 148325 return entry;
117 }
118
119 /**
120 * Check if a symbol exists in the current or any parent scope and return it if possible.
121 * If the target symbol is an alias symbol, resolve the alias container entry.
122 *
123 * @param name Name of the desired symbol
124 * @return Desired symbol / nullptr if the symbol was not found + alias or not
125 */
126 51128 std::pair<SymbolTableEntry *, bool> SymbolTable::lookupWithAliasResolution(const std::string &name) {
127
1/2
✓ Branch 2 → 3 taken 51128 times.
✗ Branch 2 → 28 not taken.
51128 SymbolTableEntry *entry = lookup(name);
128
8/10
✓ Branch 3 → 4 taken 10913 times.
✓ Branch 3 → 7 taken 40215 times.
✓ Branch 4 → 5 taken 10913 times.
✗ Branch 4 → 28 not taken.
✓ Branch 5 → 6 taken 10913 times.
✗ Branch 5 → 28 not taken.
✓ Branch 6 → 7 taken 10911 times.
✓ Branch 6 → 8 taken 2 times.
✓ Branch 9 → 10 taken 51126 times.
✓ Branch 9 → 12 taken 2 times.
51128 if (!entry || !entry->getQualType().is(TY_ALIAS))
129 51126 return {entry, false};
130
131 // We have an alias type here, resolve it
132
1/2
✗ Branch 14 → 15 not taken.
✓ Branch 14 → 16 taken 2 times.
4 assert(entry->scope->isRootScope());
133 2 entry->used = true;
134
1/2
✓ Branch 16 → 17 taken 2 times.
✗ Branch 16 → 28 not taken.
2 const std::string aliasedContainerEntryName = entry->name + ALIAS_CONTAINER_SUFFIX;
135
1/2
✓ Branch 17 → 18 taken 2 times.
✗ Branch 17 → 26 not taken.
2 SymbolTableEntry *aliasedTypeContainerEntry = entry->scope->lookupStrict(aliasedContainerEntryName);
136
1/2
✗ Branch 20 → 21 not taken.
✓ Branch 20 → 22 taken 2 times.
2 assert(aliasedTypeContainerEntry != nullptr);
137 2 return {aliasedTypeContainerEntry, true};
138 2 }
139
140 /**
141 * Check if a symbol exists in the current scope and return it if possible
142 *
143 * @param symbolName Name of the desired symbol
144 * @return Desired symbol / nullptr if the symbol was not found
145 */
146 1269647 SymbolTableEntry *SymbolTable::lookupStrict(const std::string &symbolName) {
147
1/2
✗ Branch 3 → 4 not taken.
✓ Branch 3 → 5 taken 1269647 times.
1269647 if (symbolName.empty())
148 return nullptr;
149 // Check if a symbol with this name exists in this scope
150
3/4
✓ Branch 5 → 6 taken 1269647 times.
✗ Branch 5 → 19 not taken.
✓ Branch 8 → 9 taken 673565 times.
✓ Branch 8 → 11 taken 596082 times.
1269647 if (const auto it = symbols.find(symbolName); it != symbols.end())
151 673565 return &it->second;
152 // Check if a capture with this name exists in this scope
153
3/4
✓ Branch 11 → 12 taken 596082 times.
✗ Branch 11 → 20 not taken.
✓ Branch 14 → 15 taken 25 times.
✓ Branch 14 → 17 taken 596057 times.
596082 if (const auto it = captures.find(symbolName); it != captures.end())
154 25 return it->second.capturedSymbol;
155 // Otherwise, return a nullptr
156 596057 return nullptr;
157 }
158
159 /**
160 * Check if a symbol exists in one of the composed field scopes of the current scope and return it if possible.
161 * This only works if the current scope is a struct scope.
162 *
163 * @param name Name of the desired symbol
164 * @param indexPath How to index the found symbol using order indices (e.g. for GEP)
165 * @return Desired symbol / nullptr if the symbol was not found
166 */
167 92683 SymbolTableEntry *SymbolTable::lookupInComposedFields(const std::string &name, // NOLINT(misc-no-recursion)
168 std::vector<size_t> &indexPath) {
169
1/2
✗ Branch 2 → 3 not taken.
✓ Branch 2 → 4 taken 92683 times.
92683 assert(scope->type == ScopeType::STRUCT);
170
171 // Check if we have a symbol with this name in the current scope
172
2/2
✓ Branch 5 → 6 taken 92574 times.
✓ Branch 5 → 8 taken 109 times.
92683 if (SymbolTableEntry *result = lookupStrict(name)) {
173 92574 indexPath.push_back(result->orderIndex);
174 92574 return result;
175 }
176
177 // If it was not found in the current scope, loop through all composed fields in this scope
178
2/2
✓ Branch 25 → 9 taken 121 times.
✓ Branch 25 → 26 taken 13 times.
134 for (size_t i = 0; i < scope->getFieldCount(); i++) {
179 121 const SymbolTableEntry *fieldEntry = lookupStrictByIndex(i);
180
181 // Skip all fields that are not composition fields
182
2/2
✓ Branch 12 → 13 taken 14 times.
✓ Branch 12 → 14 taken 107 times.
121 if (!fieldEntry->getQualType().isComposition())
183 14 continue;
184
185 // Add the current field's order index to the index path
186 107 indexPath.push_back(fieldEntry->orderIndex);
187
188 // Search in the composed field's body scope
189 107 Scope *searchScope = fieldEntry->getQualType().getBodyScope();
190
1/2
✗ Branch 17 → 18 not taken.
✓ Branch 17 → 19 taken 107 times.
107 assert(searchScope != nullptr);
191
2/2
✓ Branch 20 → 21 taken 96 times.
✓ Branch 20 → 22 taken 11 times.
107 if (SymbolTableEntry *result = searchScope->symbolTable.lookupInComposedFields(name, indexPath))
192 96 return result;
193
194 // Remove the current field's order index from the index path
195 11 indexPath.pop_back();
196 }
197
198 // Symbol was not found in current scope, return nullptr
199 13 return nullptr;
200 }
201
202 /**
203 * Check if an order index exists in the current or any parent scope and returns it if possible.
204 * Warning: Unlike the `lookup` method, this one doesn't consider the parent scopes
205 *
206 * @param orderIndex Order index of the desired symbol
207 * @return Desired symbol / nullptr if the symbol was not found
208 */
209 178660 SymbolTableEntry *SymbolTable::lookupStrictByIndex(unsigned int orderIndex) {
210
4/8
✓ Branch 2 → 3 taken 178660 times.
✗ Branch 2 → 14 not taken.
✓ Branch 3 → 4 taken 178660 times.
✗ Branch 3 → 14 not taken.
✓ Branch 4 → 5 taken 178660 times.
✗ Branch 4 → 14 not taken.
✓ Branch 11 → 6 taken 1928742 times.
✗ Branch 11 → 12 not taken.
1928742 for (auto &val : symbols | std::views::values) {
211
2/2
✓ Branch 7 → 8 taken 178660 times.
✓ Branch 7 → 9 taken 1750082 times.
1928742 if (val.orderIndex == orderIndex)
212 178660 return &val;
213 }
214 return nullptr;
215 }
216
217 /**
218 * Check if an anonymous symbol exists in the current scope and return it if possible
219 *
220 * @param declNode AST node where the anonymous symbol is declared
221 * @param numericSuffix Numeric suffix of the anonymous symbol
222 * @return Anonymous symbol
223 */
224 22366 SymbolTableEntry *SymbolTable::lookupAnonymous(const ASTNode *declNode, size_t numericSuffix) {
225
1/2
✓ Branch 2 → 3 taken 22366 times.
✗ Branch 2 → 27 not taken.
22366 std::stringstream name;
226
5/10
✓ Branch 3 → 4 taken 22366 times.
✗ Branch 3 → 25 not taken.
✓ Branch 4 → 5 taken 22366 times.
✗ Branch 4 → 21 not taken.
✓ Branch 5 → 6 taken 22366 times.
✗ Branch 5 → 19 not taken.
✓ Branch 6 → 7 taken 22366 times.
✗ Branch 6 → 19 not taken.
✓ Branch 7 → 8 taken 22366 times.
✗ Branch 7 → 19 not taken.
22366 name << "anon." << declNode->codeLoc.toString() << "." << reinterpret_cast<size_t>(declNode);
227
2/2
✓ Branch 9 → 10 taken 530 times.
✓ Branch 9 → 12 taken 21836 times.
22366 if (numericSuffix > 0)
228
2/4
✓ Branch 10 → 11 taken 530 times.
✗ Branch 10 → 25 not taken.
✓ Branch 11 → 12 taken 530 times.
✗ Branch 11 → 25 not taken.
530 name << "." << numericSuffix;
229
2/4
✓ Branch 12 → 13 taken 22366 times.
✗ Branch 12 → 24 not taken.
✓ Branch 13 → 14 taken 22366 times.
✗ Branch 13 → 22 not taken.
44732 return lookup(name.str());
230 22366 }
231
232 /**
233 * Check if a capture exists in the current or any parent scope and return it if possible
234 *
235 * @param name Name of the desired captured symbol
236 * @return Capture / nullptr if the capture was not found
237 */
238 556795 Capture *SymbolTable::lookupCapture(const std::string &name) { // NOLINT(misc-no-recursion)
239 // Check if the capture exists in the current scope. If yes, take it
240
2/2
✓ Branch 3 → 4 taken 81 times.
✓ Branch 3 → 5 taken 556714 times.
556795 if (Capture *capture = lookupCaptureStrict(name))
241 81 return capture;
242
243 // We reached the root scope, the symbol does not exist at all
244
2/2
✓ Branch 5 → 6 taken 163337 times.
✓ Branch 5 → 7 taken 393377 times.
556714 if (parent == nullptr)
245 163337 return nullptr;
246
247 393377 return parent->lookupCapture(name);
248 }
249
250 /**
251 * Check if a capture exists in the current scope and return it if possible
252 *
253 * @param name Name of the desired captured symbol
254 * @return Capture / nullptr if the capture was not found
255 */
256 556795 Capture *SymbolTable::lookupCaptureStrict(const std::string &name) {
257 // If available in the current scope, return it
258
1/2
✓ Branch 2 → 3 taken 556795 times.
✗ Branch 2 → 11 not taken.
556795 const auto it = captures.find(name);
259
2/2
✓ Branch 5 → 6 taken 81 times.
✓ Branch 5 → 8 taken 556714 times.
556795 if (it != captures.end())
260 81 return &it->second;
261 // Otherwise, return nullptr
262 556714 return nullptr;
263 }
264
265 /**
266 * Set capturing for this scope required.
267 */
268 63 void SymbolTable::setCapturingRequired() { capturingRequired = true; }
269
270 /**
271 * Deletes an existing anonymous symbol
272 *
273 * @param name Anonymous symbol name
274 */
275 10491 void SymbolTable::deleteAnonymous(const std::string &name) { symbols.erase(name); }
276
277 /**
278 * Stringify a symbol table to a human-readable form. This is used to realize dumps of symbol tables
279 *
280 * Example:
281 * {
282 * "name": "<SymbolTableName>"
283 * "symbols": [
284 * ... (SymbolTableEntry)
285 * ],
286 * "children": [
287 * ... (SymbolTable)
288 * ]
289 * }
290 *
291 * @return Symbol table if form of a string
292 */
293 381373 nlohmann::json SymbolTable::toJSON() const {
294 // Collect all symbols
295 381373 std::vector<nlohmann::json> jsonSymbols;
296
1/2
✓ Branch 3 → 4 taken 381373 times.
✗ Branch 3 → 59 not taken.
381373 jsonSymbols.reserve(symbols.size());
297
5/8
✓ Branch 4 → 5 taken 381373 times.
✗ Branch 4 → 44 not taken.
✓ Branch 5 → 6 taken 381373 times.
✗ Branch 5 → 44 not taken.
✓ Branch 6 → 7 taken 381373 times.
✗ Branch 6 → 44 not taken.
✓ Branch 14 → 8 taken 855685 times.
✓ Branch 14 → 15 taken 381373 times.
1237058 for (const SymbolTableEntry &symbol : symbols | std::views::values)
298
2/4
✓ Branch 9 → 10 taken 855685 times.
✗ Branch 9 → 43 not taken.
✓ Branch 10 → 11 taken 855685 times.
✗ Branch 10 → 41 not taken.
855685 jsonSymbols.emplace_back(symbol.toJSON());
299
300 // Collect all captures
301 381373 std::vector<nlohmann::json> jsonCaptures;
302
1/2
✓ Branch 16 → 17 taken 381373 times.
✗ Branch 16 → 57 not taken.
381373 jsonCaptures.reserve(captures.size());
303
5/8
✓ Branch 17 → 18 taken 381373 times.
✗ Branch 17 → 48 not taken.
✓ Branch 18 → 19 taken 381373 times.
✗ Branch 18 → 48 not taken.
✓ Branch 19 → 20 taken 381373 times.
✗ Branch 19 → 48 not taken.
✓ Branch 27 → 21 taken 81 times.
✓ Branch 27 → 28 taken 381373 times.
381454 for (const Capture &capture : captures | std::views::values)
304
2/4
✓ Branch 22 → 23 taken 81 times.
✗ Branch 22 → 47 not taken.
✓ Branch 23 → 24 taken 81 times.
✗ Branch 23 → 45 not taken.
81 jsonCaptures.emplace_back(capture.toJSON());
305
306 // Generate json
307 381373 nlohmann::json result;
308
2/4
✓ Branch 29 → 30 taken 381373 times.
✗ Branch 29 → 51 not taken.
✓ Branch 30 → 31 taken 381373 times.
✗ Branch 30 → 49 not taken.
381373 result["symbols"] = jsonSymbols;
309
2/4
✓ Branch 33 → 34 taken 381373 times.
✗ Branch 33 → 54 not taken.
✓ Branch 34 → 35 taken 381373 times.
✗ Branch 34 → 52 not taken.
381373 result["captures"] = jsonCaptures;
310 381373 return result;
311 381373 }
312
313 } // namespace spice::compiler
314