GCC Code Coverage Report


Directory: ../
Coverage: low: ≥ 0% medium: ≥ 75.0% high: ≥ 90.0%
Coverage Exec / Excl / Total
Lines: 100.0% 45 / 0 / 45
Functions: 100.0% 6 / 0 / 6
Branches: 100.0% 6 / 0 / 6

src/util/CustomHashFunctions.cpp
Line Branch Exec Source
1 // Copyright (c) 2021-2026 ChilliBits. All rights reserved.
2
3 #include "CustomHashFunctions.h"
4
5 #include <numeric>
6 #include <string>
7
8 #include <symboltablebuilder/Type.h>
9 #include <symboltablebuilder/TypeChain.h>
10
11 namespace spice::compiler {
12
13 41587194 uint64_t hashMix(uint64_t hash) noexcept {
14 41587194 hash += 0x9e3779b97f4a7c15ull;
15 41587194 hash = (hash ^ (hash >> 30)) * 0xbf58476d1ce4e5b9ull;
16 41587194 hash = (hash ^ (hash >> 27)) * 0x94d049bb133111ebull;
17 41587194 hash ^= (hash >> 31);
18 41587194 return hash;
19 }
20
21 32001931 void hashCombine64(uint64_t &hash, uint64_t value) noexcept {
22 32001931 hash ^= hashMix(value + 0x9e3779b97f4a7c15ull + (hash << 6) + (hash >> 2));
23 32001931 }
24
25 } // namespace spice::compiler
26
27 namespace std {
28
29 4037243 size_t hash<spice::compiler::TypeChainElement>::operator()(const spice::compiler::TypeChainElement &tce) const noexcept {
30 using namespace spice::compiler;
31 4037243 uint64_t hash = 0;
32
33 4037243 hashCombine64(hash, tce.superType);
34 4037243 hashCombine64(hash, std::hash<std::string>{}(tce.subType));
35 4037243 hashCombine64(hash, tce.typeId);
36
37
4/4
✓ Branch 6 → 7 taken 1008 times.
✓ Branch 6 → 9 taken 96520 times.
✓ Branch 6 → 14 taken 776261 times.
✓ Branch 6 → 17 taken 3163454 times.
4037243 switch (tce.superType) {
38 1008 case TY_ARRAY:
39 1008 hashCombine64(hash, tce.data.arraySize);
40 1008 break;
41 96520 case TY_FUNCTION:
42 case TY_PROCEDURE:
43
2/2
✓ Branch 9 → 10 taken 210 times.
✓ Branch 9 → 11 taken 96310 times.
96520 hashCombine64(hash, tce.data.hasCaptures ? 0xF00D1234ULL : 0xBAD0C0DEULL);
44 96520 break;
45 776261 case TY_STRUCT:
46 case TY_INTERFACE:
47 case TY_ENUM:
48 // Stable hash based on pointer identity, but randomized for safety
49 776261 hashCombine64(hash, hashPointer(tce.data.bodyScope));
50 776261 break;
51 3163454 default:
52 3163454 break;
53 }
54
55 4037243 hashCombine64(hash, hashVector(tce.templateTypes));
56 4037243 hashCombine64(hash, hashVector(tce.paramTypes));
57
58 4037243 return hashMix(hash);
59 }
60
61 3865582 size_t hash<spice::compiler::Type>::operator()(const spice::compiler::Type &t) const noexcept {
62 using namespace spice::compiler;
63 3865582 uint64_t hash = 0;
64 3865582 hashCombine64(hash, hashVector(t.typeChain));
65 3865582 return hashMix(hash);
66 }
67
68 841219 size_t hash<spice::compiler::TypeQualifiers>::operator()(const spice::compiler::TypeQualifiers &qualifiers) const noexcept {
69 using namespace spice::compiler;
70 841219 const uint8_t bits = (qualifiers.isConst << 0) | (qualifiers.isSigned << 1) | (qualifiers.isUnsigned << 2) |
71 841219 (qualifiers.isHeap << 3) | (qualifiers.isPublic << 4) | (qualifiers.isInline << 5) |
72 841219 (qualifiers.isComposition << 6);
73 841219 return hashMix(bits);
74 }
75
76 841219 size_t hash<spice::compiler::QualType>::operator()(const spice::compiler::QualType &qualType) const noexcept {
77 using namespace spice::compiler;
78 841219 uint64_t seed = 0;
79
80 // Hash type pointer content if possible
81 841219 hashCombine64(seed, std::hash<Type>{}(*qualType.getType()));
82
83 841219 TypeQualifiers qualifiers = qualType.getQualifiers();
84 841219 qualifiers.isPublic = false; // Ignore the public qualifier for hashing
85 841219 hashCombine64(seed, std::hash<TypeQualifiers>{}(qualifiers));
86 841219 return hashMix(seed);
87 }
88
89 } // namespace std
90