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