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 195821473 uint64_t hashMix(uint64_t hash) noexcept {
14 195821473 hash += 0x9e3779b97f4a7c15ull;
15 195821473 hash = (hash ^ (hash >> 30)) * 0xbf58476d1ce4e5b9ull;
16 195821473 hash = (hash ^ (hash >> 27)) * 0x94d049bb133111ebull;
17 195821473 hash ^= (hash >> 31);
18 195821473 return hash;
19 }
20
21 150551578 void hashCombine64(uint64_t &hash, uint64_t value) noexcept {
22 150551578 hash ^= hashMix(value + 0x9e3779b97f4a7c15ull + (hash << 6) + (hash >> 2));
23 150551578 }
24
25 } // namespace spice::compiler
26
27 namespace std {
28
29 17972734 size_t hash<spice::compiler::TypeChainElement>::operator()(const spice::compiler::TypeChainElement &tce) const noexcept {
30 using namespace spice::compiler;
31 17972734 uint64_t hash = 0;
32
33 17972734 hashCombine64(hash, tce.superType);
34 17972734 hashCombine64(hash, std::hash<std::string>{}(tce.subType));
35 17972734 hashCombine64(hash, tce.typeId);
36
37
4/4
✓ Branch 6 → 7 taken 2671 times.
✓ Branch 6 → 9 taken 404836 times.
✓ Branch 6 → 14 taken 7426287 times.
✓ Branch 6 → 17 taken 10138940 times.
17972734 switch (tce.superType) {
38 2671 case TY_ARRAY:
39 2671 hashCombine64(hash, tce.data.arraySize);
40 2671 break;
41 404836 case TY_FUNCTION:
42 case TY_PROCEDURE:
43
2/2
✓ Branch 9 → 10 taken 710 times.
✓ Branch 9 → 11 taken 404126 times.
404836 hashCombine64(hash, tce.data.hasCaptures ? 0xF00D1234ULL : 0xBAD0C0DEULL);
44 404836 break;
45 7426287 case TY_STRUCT:
46 case TY_INTERFACE:
47 case TY_ENUM:
48 // Stable hash based on pointer identity, but randomized for safety
49 7426287 hashCombine64(hash, hashPointer(tce.data.bodyScope));
50 7426287 break;
51 10138940 default:
52 10138940 break;
53 }
54
55 17972734 hashCombine64(hash, hashVector(tce.templateTypes));
56 17972734 hashCombine64(hash, hashVector(tce.paramTypes));
57
58 17972734 return hashMix(hash);
59 }
60
61 16828683 size_t hash<spice::compiler::Type>::operator()(const spice::compiler::Type &t) const noexcept {
62 using namespace spice::compiler;
63 16828683 uint64_t hash = 0;
64 16828683 hashCombine64(hash, hashVector(t.typeChain));
65 16828683 return hashMix(hash);
66 }
67
68 5234239 size_t hash<spice::compiler::TypeQualifiers>::operator()(const spice::compiler::TypeQualifiers &qualifiers) const noexcept {
69 using namespace spice::compiler;
70 5234239 const uint8_t bits = (qualifiers.isConst << 0) | (qualifiers.isSigned << 1) | (qualifiers.isUnsigned << 2) |
71 5234239 (qualifiers.isHeap << 3) | (qualifiers.isPublic << 4) | (qualifiers.isInline << 5) |
72 5234239 (qualifiers.isComposition << 6);
73 5234239 return hashMix(bits);
74 }
75
76 5234239 size_t hash<spice::compiler::QualType>::operator()(const spice::compiler::QualType &qualType) const noexcept {
77 using namespace spice::compiler;
78 5234239 uint64_t seed = 0;
79
80 // Hash type pointer content if possible
81 5234239 hashCombine64(seed, std::hash<Type>{}(*qualType.getType()));
82
83 5234239 TypeQualifiers qualifiers = qualType.getQualifiers();
84 5234239 qualifiers.isPublic = false; // Ignore the public qualifier for hashing
85 5234239 hashCombine64(seed, std::hash<TypeQualifiers>{}(qualifiers));
86 5234239 return hashMix(seed);
87 }
88
89 } // namespace std
90