Austin Eng | fd4688e | 2019-01-30 02:20:58 +0000 | [diff] [blame] | 1 | // Copyright 2019 The Dawn Authors |
| 2 | // |
| 3 | // Licensed under the Apache License, Version 2.0 (the "License"); |
| 4 | // you may not use this file except in compliance with the License. |
| 5 | // You may obtain a copy of the License at |
| 6 | // |
| 7 | // http://www.apache.org/licenses/LICENSE-2.0 |
| 8 | // |
| 9 | // Unless required by applicable law or agreed to in writing, software |
| 10 | // distributed under the License is distributed on an "AS IS" BASIS, |
| 11 | // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |
| 12 | // See the License for the specific language governing permissions and |
| 13 | // limitations under the License. |
| 14 | |
| 15 | #ifndef DAWNWIRE_SERVER_OBJECTSTORAGE_H_ |
| 16 | #define DAWNWIRE_SERVER_OBJECTSTORAGE_H_ |
| 17 | |
Austin Eng | fd4688e | 2019-01-30 02:20:58 +0000 | [diff] [blame] | 18 | #include "dawn_wire/WireCmd_autogen.h" |
Austin Eng | 6a5418a | 2019-07-19 16:01:48 +0000 | [diff] [blame] | 19 | #include "dawn_wire/WireServer.h" |
Austin Eng | fd4688e | 2019-01-30 02:20:58 +0000 | [diff] [blame] | 20 | |
| 21 | #include <algorithm> |
| 22 | #include <map> |
| 23 | |
| 24 | namespace dawn_wire { namespace server { |
| 25 | |
| 26 | template <typename T> |
| 27 | struct ObjectDataBase { |
Austin Eng | 518c8e7 | 2020-04-13 17:50:51 +0000 | [diff] [blame] | 28 | // The backend-provided handle and generation to this object. |
Austin Eng | fd4688e | 2019-01-30 02:20:58 +0000 | [diff] [blame] | 29 | T handle; |
Austin Eng | 518c8e7 | 2020-04-13 17:50:51 +0000 | [diff] [blame] | 30 | uint32_t generation = 0; |
Austin Eng | fd4688e | 2019-01-30 02:20:58 +0000 | [diff] [blame] | 31 | |
Austin Eng | fd4688e | 2019-01-30 02:20:58 +0000 | [diff] [blame] | 32 | // Whether this object has been allocated, used by the KnownObjects queries |
| 33 | // TODO(cwallez@chromium.org): make this an internal bit vector in KnownObjects. |
| 34 | bool allocated; |
| 35 | }; |
| 36 | |
| 37 | // Stores what the backend knows about the type. |
Corentin Wallez | cb2c64f | 2019-04-01 21:04:17 +0000 | [diff] [blame] | 38 | template <typename T> |
Austin Eng | fd4688e | 2019-01-30 02:20:58 +0000 | [diff] [blame] | 39 | struct ObjectData : public ObjectDataBase<T> {}; |
| 40 | |
Austin Eng | 120f5d9 | 2019-06-07 03:53:07 +0000 | [diff] [blame] | 41 | enum class BufferMapWriteState { Unmapped, Mapped, MapError }; |
| 42 | |
Austin Eng | fd4688e | 2019-01-30 02:20:58 +0000 | [diff] [blame] | 43 | template <> |
Corentin Wallez | 1fdcb16 | 2019-10-24 22:27:27 +0000 | [diff] [blame] | 44 | struct ObjectData<WGPUBuffer> : public ObjectDataBase<WGPUBuffer> { |
Austin Eng | 6a5418a | 2019-07-19 16:01:48 +0000 | [diff] [blame] | 45 | // TODO(enga): Use a tagged pointer to save space. |
| 46 | std::unique_ptr<MemoryTransferService::ReadHandle> readHandle; |
| 47 | std::unique_ptr<MemoryTransferService::WriteHandle> writeHandle; |
Austin Eng | 120f5d9 | 2019-06-07 03:53:07 +0000 | [diff] [blame] | 48 | BufferMapWriteState mapWriteState = BufferMapWriteState::Unmapped; |
Austin Eng | fd4688e | 2019-01-30 02:20:58 +0000 | [diff] [blame] | 49 | }; |
| 50 | |
| 51 | // Keeps track of the mapping between client IDs and backend objects. |
| 52 | template <typename T> |
| 53 | class KnownObjects { |
| 54 | public: |
| 55 | using Data = ObjectData<T>; |
| 56 | |
| 57 | KnownObjects() { |
Corentin Wallez | b92fa10 | 2019-02-25 09:49:00 +0000 | [diff] [blame] | 58 | // Reserve ID 0 so that it can be used to represent nullptr for optional object values |
| 59 | // in the wire format. However don't tag it as allocated so that it is an error to ask |
| 60 | // KnownObjects for ID 0. |
| 61 | Data reservation; |
| 62 | reservation.handle = nullptr; |
Corentin Wallez | b92fa10 | 2019-02-25 09:49:00 +0000 | [diff] [blame] | 63 | reservation.allocated = false; |
Austin Eng | 6a5418a | 2019-07-19 16:01:48 +0000 | [diff] [blame] | 64 | mKnown.push_back(std::move(reservation)); |
Austin Eng | fd4688e | 2019-01-30 02:20:58 +0000 | [diff] [blame] | 65 | } |
| 66 | |
| 67 | // Get a backend objects for a given client ID. |
| 68 | // Returns nullptr if the ID hasn't previously been allocated. |
| 69 | const Data* Get(uint32_t id) const { |
| 70 | if (id >= mKnown.size()) { |
| 71 | return nullptr; |
| 72 | } |
| 73 | |
| 74 | const Data* data = &mKnown[id]; |
| 75 | |
| 76 | if (!data->allocated) { |
| 77 | return nullptr; |
| 78 | } |
| 79 | |
| 80 | return data; |
| 81 | } |
| 82 | Data* Get(uint32_t id) { |
| 83 | if (id >= mKnown.size()) { |
| 84 | return nullptr; |
| 85 | } |
| 86 | |
| 87 | Data* data = &mKnown[id]; |
| 88 | |
| 89 | if (!data->allocated) { |
| 90 | return nullptr; |
| 91 | } |
| 92 | |
| 93 | return data; |
| 94 | } |
| 95 | |
| 96 | // Allocates the data for a given ID and returns it. |
Natasha Lee | 17a8498 | 2020-02-25 19:20:15 +0000 | [diff] [blame] | 97 | // Returns nullptr if the ID is already allocated, or too far ahead, or if ID is 0 (ID 0 is |
| 98 | // reserved for nullptr). Invalidates all the Data* |
Austin Eng | fd4688e | 2019-01-30 02:20:58 +0000 | [diff] [blame] | 99 | Data* Allocate(uint32_t id) { |
Natasha Lee | 17a8498 | 2020-02-25 19:20:15 +0000 | [diff] [blame] | 100 | if (id == 0 || id > mKnown.size()) { |
Austin Eng | fd4688e | 2019-01-30 02:20:58 +0000 | [diff] [blame] | 101 | return nullptr; |
| 102 | } |
| 103 | |
| 104 | Data data; |
| 105 | data.allocated = true; |
Austin Eng | fd4688e | 2019-01-30 02:20:58 +0000 | [diff] [blame] | 106 | data.handle = nullptr; |
| 107 | |
| 108 | if (id >= mKnown.size()) { |
Austin Eng | 6a5418a | 2019-07-19 16:01:48 +0000 | [diff] [blame] | 109 | mKnown.push_back(std::move(data)); |
Austin Eng | fd4688e | 2019-01-30 02:20:58 +0000 | [diff] [blame] | 110 | return &mKnown.back(); |
| 111 | } |
| 112 | |
| 113 | if (mKnown[id].allocated) { |
| 114 | return nullptr; |
| 115 | } |
| 116 | |
Austin Eng | 6a5418a | 2019-07-19 16:01:48 +0000 | [diff] [blame] | 117 | mKnown[id] = std::move(data); |
Austin Eng | fd4688e | 2019-01-30 02:20:58 +0000 | [diff] [blame] | 118 | return &mKnown[id]; |
| 119 | } |
| 120 | |
| 121 | // Marks an ID as deallocated |
| 122 | void Free(uint32_t id) { |
| 123 | ASSERT(id < mKnown.size()); |
| 124 | mKnown[id].allocated = false; |
| 125 | } |
| 126 | |
| 127 | std::vector<T> AcquireAllHandles() { |
| 128 | std::vector<T> objects; |
| 129 | for (Data& data : mKnown) { |
| 130 | if (data.allocated && data.handle != nullptr) { |
| 131 | objects.push_back(data.handle); |
Austin Eng | fd4688e | 2019-01-30 02:20:58 +0000 | [diff] [blame] | 132 | data.allocated = false; |
| 133 | data.handle = nullptr; |
| 134 | } |
| 135 | } |
| 136 | |
| 137 | return objects; |
| 138 | } |
| 139 | |
| 140 | private: |
| 141 | std::vector<Data> mKnown; |
| 142 | }; |
| 143 | |
| 144 | // ObjectIds are lost in deserialization. Store the ids of deserialized |
| 145 | // objects here so they can be used in command handlers. This is useful |
| 146 | // for creating ReturnWireCmds which contain client ids |
| 147 | template <typename T> |
| 148 | class ObjectIdLookupTable { |
| 149 | public: |
| 150 | void Store(T key, ObjectId id) { |
| 151 | mTable[key] = id; |
| 152 | } |
| 153 | |
| 154 | // Return the cached ObjectId, or 0 (null handle) |
| 155 | ObjectId Get(T key) const { |
| 156 | const auto it = mTable.find(key); |
| 157 | if (it != mTable.end()) { |
| 158 | return it->second; |
| 159 | } |
| 160 | return 0; |
| 161 | } |
| 162 | |
| 163 | void Remove(T key) { |
| 164 | auto it = mTable.find(key); |
| 165 | if (it != mTable.end()) { |
| 166 | mTable.erase(it); |
| 167 | } |
| 168 | } |
| 169 | |
| 170 | private: |
| 171 | std::map<T, ObjectId> mTable; |
| 172 | }; |
| 173 | |
| 174 | }} // namespace dawn_wire::server |
| 175 | |
| 176 | #endif // DAWNWIRE_SERVER_OBJECTSTORAGE_H_ |