File: sync_access_flags.h

package info (click to toggle)
vulkan-validationlayers 1.4.335.0-2
  • links: PTS, VCS
  • area: main
  • in suites: sid
  • size: 51,728 kB
  • sloc: cpp: 645,254; python: 12,203; sh: 24; makefile: 24; xml: 14
file content (119 lines) | stat: -rw-r--r-- 4,167 bytes parent folder | download | duplicates (4)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
/* Copyright (c) 2025 The Khronos Group Inc.
 * Copyright (c) 2025 Valve Corporation
 * Copyright (c) 2025 LunarG, Inc.
 *
 * Licensed under the Apache License, Version 2.0 (the "License");
 * you may not use this file except in compliance with the License.
 * You may obtain a copy of the License at
 *
 *     http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 */

#pragma once

#include "utils/hash_util.h"
#include <cstring>  // memset

namespace syncval {

// NOTE: SyncAccessFlags struct replaces what was originally defined via std::bitset<>.
// This is done for simplicity and efficiency purposes. The general nature of std::bitset
// makes it inefficient in some scenarios (e.g. hashing). Also implementation simplicity
// is important because SyncAccessFlags is one of the main types in syncval implementation.

struct SyncAccessFlags {
    // Syncval currently requires more than 128 bits to store access flags.
    // We use three 64-bit words (192 bits) for storage.
    static constexpr size_t kWordCount = 3;
    static constexpr size_t kBitCount = kWordCount * sizeof(uint64_t) * 8;

    uint64_t words[kWordCount];

    SyncAccessFlags() {
        // Memset generates the most efficient code for specific arch
        std::memset(words, 0, sizeof(words));
    }

    SyncAccessFlags(uint64_t value) {
        words[0] = value;
        words[1] = 0;
        words[2] = 0;
    }

    SyncAccessFlags(uint64_t word0, uint64_t word1, uint64_t word2) {
        words[0] = word0;
        words[1] = word1;
        words[2] = word2;
    }

    void reset() { std::memset(words, 0, sizeof(words)); }

    // NOTE: In this implementation, (pos >> 6) is equivalent to (pos / 64) and gives the
    // index of the 64-bit word. Then (pos & 63u) gives the bit index within that word.

    void reset(size_t pos) {
        assert(pos < kBitCount);
        words[pos >> 6] &= ~(uint64_t(1) << (pos & 63u));
    }

    void set(size_t pos) {
        assert(pos < kBitCount);
        words[pos >> 6] |= (uint64_t(1) << (pos & 63u));
    }

    SyncAccessFlags& operator|=(const SyncAccessFlags& other) {
        words[0] |= other.words[0];
        words[1] |= other.words[1];
        words[2] |= other.words[2];
        return *this;
    }

    SyncAccessFlags operator|(const SyncAccessFlags& other) const {
        return SyncAccessFlags((words[0] | other.words[0]), (words[1] | other.words[1]), (words[2] | other.words[2]));
    }

    SyncAccessFlags operator&(const SyncAccessFlags& other) const {
        return SyncAccessFlags((words[0] & other.words[0]), (words[1] & other.words[1]), (words[2] & other.words[2]));
    }

    SyncAccessFlags operator~() const { return SyncAccessFlags(~words[0], ~words[1], ~words[2]); }

    SyncAccessFlags operator<<(size_t pos) const {
        assert(pos < kBitCount);

        // The assert shows the only use case used by syncval.
        // This general operator will be replaced by specific function and assert won't be needed.
        assert(words[0] == 1 && words[1] == 0 && words[2] == 0);

        uint64_t words[3] = {};
        words[pos >> 6] = (uint64_t(1) << (pos & 63u));
        return SyncAccessFlags(words[0], words[1], words[2]);
    }

    bool operator==(const SyncAccessFlags& other) const {
        return words[0] == other.words[0] && words[1] == other.words[1] && words[2] == other.words[2];
    }

    bool operator[](size_t pos) const {
        assert(pos < kBitCount);
        return (words[pos >> 6] & (uint64_t(1) << (pos & 63u))) != 0;
    }

    bool none() const { return words[0] == 0 && words[1] == 0 && words[2] == 0; }
    bool any() const { return !none(); }
    size_t size() const { return kBitCount; }

    void HashCombine(hash_util::HashCombiner& hc) const {
        hc << words[0];
        hc << words[1];
        hc << words[2];
    }
};

}  // namespace syncval