blob: 99631586843f6b19cf4eb667f91d77514aad1a4b [file] [log] [blame]
Austin Engfd4688e2019-01-30 02:20:58 +00001// 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 Engfd4688e2019-01-30 02:20:58 +000018#include "dawn_wire/WireCmd_autogen.h"
Austin Eng6a5418a2019-07-19 16:01:48 +000019#include "dawn_wire/WireServer.h"
Austin Engfd4688e2019-01-30 02:20:58 +000020
21#include <algorithm>
22#include <map>
23
24namespace dawn_wire { namespace server {
25
26 template <typename T>
27 struct ObjectDataBase {
Austin Eng518c8e72020-04-13 17:50:51 +000028 // The backend-provided handle and generation to this object.
Austin Engfd4688e2019-01-30 02:20:58 +000029 T handle;
Austin Eng518c8e72020-04-13 17:50:51 +000030 uint32_t generation = 0;
Austin Engfd4688e2019-01-30 02:20:58 +000031
Austin Engfd4688e2019-01-30 02:20:58 +000032 // 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 Wallezcb2c64f2019-04-01 21:04:17 +000038 template <typename T>
Austin Engfd4688e2019-01-30 02:20:58 +000039 struct ObjectData : public ObjectDataBase<T> {};
40
Austin Eng120f5d92019-06-07 03:53:07 +000041 enum class BufferMapWriteState { Unmapped, Mapped, MapError };
42
Austin Engfd4688e2019-01-30 02:20:58 +000043 template <>
Corentin Wallez1fdcb162019-10-24 22:27:27 +000044 struct ObjectData<WGPUBuffer> : public ObjectDataBase<WGPUBuffer> {
Austin Eng6a5418a2019-07-19 16:01:48 +000045 // TODO(enga): Use a tagged pointer to save space.
46 std::unique_ptr<MemoryTransferService::ReadHandle> readHandle;
47 std::unique_ptr<MemoryTransferService::WriteHandle> writeHandle;
Austin Eng120f5d92019-06-07 03:53:07 +000048 BufferMapWriteState mapWriteState = BufferMapWriteState::Unmapped;
Austin Engfd4688e2019-01-30 02:20:58 +000049 };
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 Wallezb92fa102019-02-25 09:49:00 +000058 // 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 Wallezb92fa102019-02-25 09:49:00 +000063 reservation.allocated = false;
Austin Eng6a5418a2019-07-19 16:01:48 +000064 mKnown.push_back(std::move(reservation));
Austin Engfd4688e2019-01-30 02:20:58 +000065 }
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 Lee17a84982020-02-25 19:20:15 +000097 // 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 Engfd4688e2019-01-30 02:20:58 +000099 Data* Allocate(uint32_t id) {
Natasha Lee17a84982020-02-25 19:20:15 +0000100 if (id == 0 || id > mKnown.size()) {
Austin Engfd4688e2019-01-30 02:20:58 +0000101 return nullptr;
102 }
103
104 Data data;
105 data.allocated = true;
Austin Engfd4688e2019-01-30 02:20:58 +0000106 data.handle = nullptr;
107
108 if (id >= mKnown.size()) {
Austin Eng6a5418a2019-07-19 16:01:48 +0000109 mKnown.push_back(std::move(data));
Austin Engfd4688e2019-01-30 02:20:58 +0000110 return &mKnown.back();
111 }
112
113 if (mKnown[id].allocated) {
114 return nullptr;
115 }
116
Austin Eng6a5418a2019-07-19 16:01:48 +0000117 mKnown[id] = std::move(data);
Austin Engfd4688e2019-01-30 02:20:58 +0000118 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 Engfd4688e2019-01-30 02:20:58 +0000132 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_