GCC Code Coverage Report


Directory: ../
Coverage: low: ≥ 0% medium: ≥ 75.0% high: ≥ 90.0%
Coverage Exec / Excl / Total
Lines: 100.0% 74 / 6 / 80
Functions: 100.0% 7 / 0 / 7
Branches: 83.3% 55 / 12 / 78

src/symboltablebuilder/TypeQualifiers.cpp
Line Branch Exec Source
1 // Copyright (c) 2021-2026 ChilliBits. All rights reserved.
2
3 #include "TypeQualifiers.h"
4
5 #include <exception/CompilerError.h>
6 #include <symboltablebuilder/TypeChain.h>
7
8 namespace spice::compiler {
9
10 /**
11 * Get default type qualifiers for a given super type
12 *
13 * @param superType Super type
14 * @return Default type qualifiers
15 */
16 5825922 TypeQualifiers TypeQualifiers::of(uint16_t superType) {
17
5/6
✓ Branch 2 → 3 taken 504257 times.
✓ Branch 2 → 5 taken 1510450 times.
✓ Branch 2 → 7 taken 398755 times.
✓ Branch 2 → 9 taken 12093 times.
✓ Branch 2 → 11 taken 3400367 times.
✗ Branch 2 → 13 not taken.
5825922 switch (superType) {
18 504257 case TY_DOUBLE: // fall-through
19 case TY_INT: // fall-through
20 case TY_SHORT: // fall-through
21 case TY_LONG:
22 504257 return {/*const*/ false, /*signed*/ true, /*unsigned*/ false};
23 1510450 case TY_BYTE: // fall-through
24 case TY_CHAR: // fall-through
25 case TY_STRING: // fall-through
26 case TY_BOOL: // fall-through
27 case TY_PTR: // fall-through
28 case TY_REF: // fall-through
29 case TY_ARRAY: // fall-through
30 case TY_STRUCT: // fall-through
31 case TY_INTERFACE: // fall-through
32 case TY_FUNCTION: // fall-through
33 case TY_PROCEDURE:
34 1510450 return {/*const*/ false, /*signed*/ false, /*unsigned*/ true};
35 398755 case TY_GENERIC:
36 // Generics must be non-signed and non-unsigned at the same time to ensure a proper function matching
37 398755 return {/*const*/ false, /*signed*/ false, /*unsigned*/ false};
38 12093 case TY_ENUM: // fall-through
39 case TY_ALIAS: // fall-through
40 case TY_IMPORT:
41 12093 return {/*const*/ true, /*signed*/ false, /*unsigned*/ true};
42 3400367 case TY_DYN: // fall-through
43 case TY_INVALID: // fall-through
44 case TY_UNRESOLVED:
45 // Return all-false qualifiers to not match anything
46 3400367 return {/*const*/ false, /*signed*/ false, /*unsigned*/ false};
47 default: // GCOV_EXCL_LINE
48 throw CompilerError(UNHANDLED_BRANCH, "Symbol qualifier fallthrough"); // GCOV_EXCL_LINE
49 }
50 }
51
52 /**
53 * Merge two sets of type qualifiers. If possible, prefer the opposite of the default of the super type
54 *
55 * @param other Other type qualifiers object
56 * @return Merged qualifiers object
57 */
58 81770 TypeQualifiers TypeQualifiers::merge(const TypeQualifiers &other) const {
59 81770 TypeQualifiers result;
60
6/8
✓ Branch 2 → 3 taken 81770 times.
✗ Branch 2 → 23 not taken.
✓ Branch 3 → 4 taken 79179 times.
✓ Branch 3 → 7 taken 2591 times.
✓ Branch 4 → 5 taken 79179 times.
✗ Branch 4 → 23 not taken.
✓ Branch 5 → 6 taken 71827 times.
✓ Branch 5 → 7 taken 7352 times.
81770 const bool isGeneric = !getBit(BIT_INDEX_SIGNED) && !getBit(BIT_INDEX_UNSIGNED);
61
2/2
✓ Branch 20 → 9 taken 572390 times.
✓ Branch 20 → 21 taken 81770 times.
654160 for (uint8_t i = 0; i <= BIT_INDEX_MAX; i++) {
62
1/2
✓ Branch 9 → 10 taken 572390 times.
✗ Branch 9 → 23 not taken.
572390 const bool x = getBit(i);
63
1/2
✓ Branch 10 → 11 taken 572390 times.
✗ Branch 10 → 23 not taken.
572390 const bool y = other.getBit(i);
64
65
4/4
✓ Branch 11 → 12 taken 490620 times.
✓ Branch 11 → 13 taken 81770 times.
✓ Branch 12 → 13 taken 81770 times.
✓ Branch 12 → 18 taken 408850 times.
572390 if (i == BIT_INDEX_SIGNED || i == BIT_INDEX_UNSIGNED) {
66
3/4
✓ Branch 13 → 14 taken 143654 times.
✓ Branch 13 → 15 taken 19886 times.
✓ Branch 16 → 17 taken 163540 times.
✗ Branch 16 → 23 not taken.
163540 result.setBit(i, isGeneric ? y : x);
67 } else {
68
1/2
✓ Branch 18 → 19 taken 408850 times.
✗ Branch 18 → 23 not taken.
408850 result.setBit(i, x | y);
69 }
70 }
71 81770 return result;
72 }
73
74 /**
75 * Check if two sets of type qualifiers match
76 *
77 * @param other The rhs qualifiers
78 * @param allowConstify Match when the types are the same, but the lhs type is more const restrictive than the rhs type
79 * @return Matching or not
80 */
81 61967 bool TypeQualifiers::match(TypeQualifiers other, bool allowConstify) const {
82 61967 const TypeQualifiers thisQualifiers = *this;
83
84 // If allowConstify is enabled, only allow to match lhs=const and rhs=non-const
85
4/4
✓ Branch 2 → 3 taken 61952 times.
✓ Branch 2 → 5 taken 15 times.
✓ Branch 3 → 4 taken 7994 times.
✓ Branch 3 → 5 taken 53958 times.
61967 if (allowConstify && thisQualifiers.isConst)
86 7994 other.isConst = true;
87
88 // Check if qualifiers are equal
89 61967 return thisQualifiers == other;
90 }
91
92 /**
93 * Erase all qualifiers that are set in the mask. This is used in type matching.
94 *
95 * @param mask Bitmask to erase with
96 */
97 23155 void TypeQualifiers::eraseWithMask(const TypeQualifiers &mask) {
98 // Zero out all bits that are set in the mask
99
2/2
✓ Branch 11 → 3 taken 162085 times.
✓ Branch 11 → 12 taken 23155 times.
185240 for (uint8_t i = 0; i <= BIT_INDEX_MAX; i++) {
100
2/2
✓ Branch 4 → 5 taken 36299 times.
✓ Branch 4 → 10 taken 125786 times.
162085 if (mask.getBit(i)) {
101 // Zero out the bit
102 36299 setBit(i, false);
103
104 // If we set the signed/unsigned bit to zero, we need to set the other to one
105
2/2
✓ Branch 6 → 7 taken 5741 times.
✓ Branch 6 → 8 taken 30558 times.
36299 if (i == BIT_INDEX_SIGNED) {
106 5741 setBit(BIT_INDEX_UNSIGNED, true);
107
2/2
✓ Branch 8 → 9 taken 15712 times.
✓ Branch 8 → 10 taken 14846 times.
30558 } else if (i == BIT_INDEX_UNSIGNED) {
108 15712 setBit(BIT_INDEX_SIGNED, true);
109 }
110 }
111 }
112 23155 }
113
114 62383 bool operator==(const TypeQualifiers &lhs, const TypeQualifiers &rhs) {
115 62383 const bool isConst = lhs.isConst == rhs.isConst;
116 62383 const bool isSigned = lhs.isSigned == rhs.isSigned;
117 62383 const bool isUnsigned = lhs.isUnsigned == rhs.isUnsigned;
118 62383 const bool isHeap = lhs.isHeap == rhs.isHeap;
119
6/8
✓ Branch 2 → 3 taken 62288 times.
✓ Branch 2 → 7 taken 95 times.
✓ Branch 3 → 4 taken 62288 times.
✗ Branch 3 → 7 not taken.
✓ Branch 4 → 5 taken 62288 times.
✗ Branch 4 → 7 not taken.
✓ Branch 5 → 6 taken 61873 times.
✓ Branch 5 → 7 taken 415 times.
62383 return isConst && isSigned && isUnsigned && isHeap;
120 }
121
122 1467814 bool TypeQualifiers::getBit(uint8_t index) const {
123
7/8
✓ Branch 2 → 3 taken 186695 times.
✓ Branch 2 → 4 taken 268465 times.
✓ Branch 2 → 5 taken 265874 times.
✓ Branch 2 → 6 taken 186695 times.
✓ Branch 2 → 7 taken 186695 times.
✓ Branch 2 → 8 taken 186695 times.
✓ Branch 2 → 9 taken 186695 times.
✗ Branch 2 → 10 not taken.
1467814 switch (index) {
124 186695 case BIT_INDEX_CONST:
125 186695 return isConst;
126 268465 case BIT_INDEX_SIGNED:
127 268465 return isSigned;
128 265874 case BIT_INDEX_UNSIGNED:
129 265874 return isUnsigned;
130 186695 case BIT_INDEX_HEAP:
131 186695 return isHeap;
132 186695 case BIT_INDEX_PUBLIC:
133 186695 return isPublic;
134 186695 case BIT_INDEX_INLINE:
135 186695 return isInline;
136 186695 case BIT_INDEX_COMPOSITION:
137 186695 return isComposition;
138 default: // GCOV_EXCL_LINE
139 throw CompilerError(UNHANDLED_BRANCH, "Bit index fallthrough"); // GCOV_EXCL_LINE
140 }
141 }
142
143 630142 bool TypeQualifiers::setBit(uint8_t index, bool value) {
144
7/8
✓ Branch 2 → 3 taken 85702 times.
✓ Branch 2 → 4 taken 103223 times.
✓ Branch 2 → 5 taken 103223 times.
✓ Branch 2 → 6 taken 82112 times.
✓ Branch 2 → 7 taken 92342 times.
✓ Branch 2 → 8 taken 81770 times.
✓ Branch 2 → 9 taken 81770 times.
✗ Branch 2 → 10 not taken.
630142 switch (index) {
145 85702 case BIT_INDEX_CONST:
146 85702 return isConst = value;
147 103223 case BIT_INDEX_SIGNED:
148 103223 return isSigned = value;
149 103223 case BIT_INDEX_UNSIGNED:
150 103223 return isUnsigned = value;
151 82112 case BIT_INDEX_HEAP:
152 82112 return isHeap = value;
153 92342 case BIT_INDEX_PUBLIC:
154 92342 return isPublic = value;
155 81770 case BIT_INDEX_INLINE:
156 81770 return isInline = value;
157 81770 case BIT_INDEX_COMPOSITION:
158 81770 return isComposition = value;
159 default: // GCOV_EXCL_LINE
160 throw CompilerError(UNHANDLED_BRANCH, "Bit index fallthrough"); // GCOV_EXCL_LINE
161 }
162 }
163
164 } // namespace spice::compiler
165