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 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 219 220 221 222 223 224 225 226 227 228 229 230 231 232 233 234 235 236 237 238 239 240 241 242 243 244 245 246 247 248 249 250 251 252 253 254 255 256 257 258 259 260 261 262 263 264 265 266 267 268 269 270 271 272 273 274 275 276 277 278 279 280 281 282 283 284 285 286 287 288 289 290 291 292 293 294 295 296 297 298 299 300 301 302 303 304 305 306 307 308 309 310 311 312 313
|
/* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 2 -*- */
/* This Source Code Form is subject to the terms of the Mozilla Public
* License, v. 2.0. If a copy of the MPL was not distributed with this
* file, You can obtain one at http://mozilla.org/MPL/2.0/. */
#ifndef WEBGPU_PARENT_H_
#define WEBGPU_PARENT_H_
#include <unordered_map>
#include "WebGPUTypes.h"
#include "base/timer.h"
#include "mozilla/WeakPtr.h"
#include "mozilla/ipc/SharedMemoryHandle.h"
#include "mozilla/webgpu/ExternalTexture.h"
#include "mozilla/webgpu/PWebGPUParent.h"
#include "mozilla/webgpu/ffi/wgpu.h"
#include "mozilla/webrender/WebRenderAPI.h"
namespace mozilla {
namespace layers {
class RemoteTextureOwnerClient;
} // namespace layers
namespace webgpu {
class SharedTexture;
class PresentationData;
// A fixed-capacity buffer for receiving textual error messages from
// `wgpu_bindings`.
//
// The `ToFFI` method returns an `ffi::WGPUErrorBuffer` pointing to our
// buffer, for you to pass to fallible FFI-visible `wgpu_bindings`
// functions. These indicate failure by storing an error message in the
// buffer, which you can retrieve by calling `GetError`.
//
// If you call `ToFFI` on this type, you must also call `GetError` to check for
// an error. Otherwise, the destructor asserts.
//
// TODO: refactor this to avoid stack-allocating the buffer all the time.
class ErrorBuffer {
// if the message doesn't fit, it will be truncated
static constexpr unsigned BUFFER_SIZE = 512;
ffi::WGPUErrorBufferType mType = ffi::WGPUErrorBufferType_None;
char mMessageUtf8[BUFFER_SIZE] = {};
bool mAwaitingGetError = false;
RawId mDeviceId = 0;
public:
ErrorBuffer();
ErrorBuffer(const ErrorBuffer&) = delete;
~ErrorBuffer();
ffi::WGPUErrorBuffer ToFFI();
ffi::WGPUErrorBufferType GetType();
static Maybe<dom::GPUErrorFilter> ErrorTypeToFilterType(
ffi::WGPUErrorBufferType aType);
struct Error {
dom::GPUErrorFilter type;
bool isDeviceLost;
nsCString message;
RawId deviceId;
};
// Retrieve the error message was stored in this buffer. Asserts that
// this instance actually contains an error (viz., that `GetType() !=
// ffi::WGPUErrorBufferType_None`).
//
// Mark this `ErrorBuffer` as having been handled, so its destructor
// won't assert.
Maybe<Error> GetError();
void CoerceValidationToInternal();
};
// Destroy/Drop messages:
// - Messages with "Destroy" in their name request deallocation of resources
// owned by the
// object and put the object in a destroyed state without deleting the object.
// It is still safe to reffer to these objects.
// - Messages with "Drop" in their name can be thought of as C++ destructors.
// They completely
// delete the object, so future attempts at accessing to these objects will
// crash. The child process should *never* send a Drop message if it still
// holds references to the object. An object that has been destroyed still
// needs to be dropped when the last reference to it dies on the child
// process.
class WebGPUParent final : public PWebGPUParent, public SupportsWeakPtr {
NS_INLINE_DECL_THREADSAFE_REFCOUNTING(WebGPUParent, override)
public:
explicit WebGPUParent(const dom::ContentParentId& aContentId);
void PostAdapterRequestDevice(RawId aDeviceId);
void BufferUnmap(RawId aDeviceId, RawId aBufferId, bool aFlush);
ipc::IPCResult RecvMessages(uint32_t nrOfMessages,
ipc::ByteBuf&& aSerializedMessages,
nsTArray<ipc::ByteBuf>&& aDataBuffers,
nsTArray<MutableSharedMemoryHandle>&& aShmems);
ipc::IPCResult RecvCreateExternalTextureSource(
RawId aDeviceId, RawId aQueueId, RawId aExternalTextureSourceId,
const ExternalTextureSourceDescriptor& aDesc);
void QueueSubmit(RawId aQueueId, RawId aDeviceId,
Span<const RawId> aCommandBuffers,
Span<const RawId> aTextureIds,
Span<const RawId> aExternalTextureSourceIds);
void DeviceCreateSwapChain(RawId aDeviceId, RawId aQueueId,
const layers::RGBDescriptor& aDesc,
const nsTArray<RawId>& aBufferIds,
const layers::RemoteTextureOwnerId& aOwnerId,
bool aUseSharedTextureInSwapChain);
void SwapChainPresent(RawId aTextureId, RawId aCommandEncoderId,
RawId aCommandBufferId,
const layers::RemoteTextureId& aRemoteTextureId,
const layers::RemoteTextureOwnerId& aOwnerId);
void SwapChainDrop(const layers::RemoteTextureOwnerId& aOwnerId,
layers::RemoteTextureTxnType aTxnType,
layers::RemoteTextureTxnId aTxnId);
void DevicePushErrorScope(RawId aDeviceId, dom::GPUErrorFilter);
PopErrorScopeResult DevicePopErrorScope(RawId aDeviceId);
ipc::IPCResult GetFrontBufferSnapshot(
IProtocol* aProtocol, const layers::RemoteTextureOwnerId& aOwnerId,
const RawId& aCommandEncoderId, const RawId& aCommandBufferId,
Maybe<Shmem>& aShmem, gfx::IntSize& aSize, uint32_t& aByteStride);
void ActorDestroy(ActorDestroyReason aWhy) override;
struct BufferMapData {
ipc::SharedMemoryMapping mShmem;
// True if buffer's usage has MAP_READ or MAP_WRITE set.
bool mHasMapFlags;
uint64_t mMappedOffset;
uint64_t mMappedSize;
RawId mDeviceId;
};
BufferMapData* GetBufferMapData(RawId aBufferId);
bool UseSharedTextureForSwapChain(ffi::WGPUSwapChainId aSwapChainId);
void DisableSharedTextureForSwapChain(ffi::WGPUSwapChainId aSwapChainId);
bool EnsureSharedTextureForSwapChain(ffi::WGPUSwapChainId aSwapChainId,
ffi::WGPUDeviceId aDeviceId,
ffi::WGPUTextureId aTextureId,
uint32_t aWidth, uint32_t aHeight,
struct ffi::WGPUTextureFormat aFormat,
ffi::WGPUTextureUsages aUsage);
void EnsureSharedTextureForReadBackPresent(
ffi::WGPUSwapChainId aSwapChainId, ffi::WGPUDeviceId aDeviceId,
ffi::WGPUTextureId aTextureId, uint32_t aWidth, uint32_t aHeight,
struct ffi::WGPUTextureFormat aFormat, ffi::WGPUTextureUsages aUsage);
std::shared_ptr<SharedTexture> CreateSharedTexture(
const layers::RemoteTextureOwnerId& aOwnerId, ffi::WGPUDeviceId aDeviceId,
ffi::WGPUTextureId aTextureId, uint32_t aWidth, uint32_t aHeight,
const struct ffi::WGPUTextureFormat aFormat,
ffi::WGPUTextureUsages aUsage);
std::shared_ptr<SharedTexture> GetSharedTexture(ffi::WGPUTextureId aId);
void PostSharedTexture(const std::shared_ptr<SharedTexture>&& aSharedTexture,
const layers::RemoteTextureId aRemoteTextureId,
const layers::RemoteTextureOwnerId aOwnerId);
bool ForwardError(ErrorBuffer& aError);
ffi::WGPUGlobal* GetContext() const { return mContext.get(); }
bool IsDeviceActive(const RawId aDeviceId) {
return mActiveDeviceIds.Contains(aDeviceId);
}
RefPtr<gfx::FileHandleWrapper> GetDeviceFenceHandle(const RawId aDeviceId);
void RemoveSharedTexture(RawId aTextureId);
const ExternalTextureSourceHost& GetExternalTextureSource(
ffi::WGPUExternalTextureSourceId aId) const;
void DestroyExternalTextureSource(RawId aId);
void DropExternalTextureSource(RawId aId);
void DeallocBufferShmem(RawId aBufferId);
void PreDeviceDrop(RawId aDeviceId);
#if defined(XP_WIN)
static Maybe<ffi::WGPUFfiLUID> GetCompositorDeviceLuid();
#endif
struct MapRequest {
WeakPtr<WebGPUParent> mParent;
ffi::WGPUDeviceId mDeviceId;
ffi::WGPUBufferId mBufferId;
ffi::WGPUHostMap mHostMap;
uint64_t mOffset;
uint64_t mSize;
};
static void MapCallback(/* std::unique_ptr<MapRequest> */ uint8_t* aUserData,
ffi::WGPUBufferMapAsyncStatus aStatus);
struct OnSubmittedWorkDoneRequest {
WeakPtr<WebGPUParent> mParent;
ffi::WGPUDeviceId mQueueId;
};
static void OnSubmittedWorkDoneCallback(
/* std::unique_ptr<OnSubmittedWorkDoneRequest> */ uint8_t* userdata);
void ReportError(RawId aDeviceId, GPUErrorFilter, const nsCString& message);
nsTArray<Maybe<ipc::shared_memory::MutableMapping>> mTempMappings;
/// A map from wgpu buffer ids to data about their shared memory segments.
/// Includes entries about mappedAtCreation, MAP_READ and MAP_WRITE buffers,
/// regardless of their state.
std::unordered_map<RawId, BufferMapData> mSharedMemoryMap;
const dom::ContentParentId mContentId;
private:
static void DeviceLostCallback(uint8_t* aUserData, uint8_t aReason,
const char* aMessage);
virtual ~WebGPUParent();
void MaintainDevices();
void LoseDevice(const RawId aDeviceId, uint8_t aReason,
const nsACString& aMessage);
UniquePtr<ffi::WGPUGlobal> mContext;
base::RepeatingTimer<WebGPUParent> mTimer;
/// Associated presentation data for each swapchain.
std::unordered_map<layers::RemoteTextureOwnerId, RefPtr<PresentationData>,
layers::RemoteTextureOwnerId::HashFn>
mPresentationDataMap;
RefPtr<layers::RemoteTextureOwnerClient> mRemoteTextureOwner;
/// Associated stack of error scopes for each device.
std::unordered_map<uint64_t, std::vector<ErrorScope>>
mErrorScopeStackByDevice;
std::unordered_map<ffi::WGPUTextureId, std::shared_ptr<SharedTexture>>
mSharedTextures;
std::unordered_map<RawId, ExternalTextureSourceHost> mExternalTextureSources;
// Store a set of DeviceIds that have been SendDeviceLost. We use this to
// limit each Device to one DeviceLost message.
nsTHashSet<RawId> mLostDeviceIds;
// Store active DeviceIds
nsTHashSet<RawId> mActiveDeviceIds;
// Shared handle of wgpu device's fence.
std::unordered_map<RawId, RefPtr<gfx::FileHandleWrapper>> mDeviceFenceHandles;
};
#if defined(XP_LINUX) && !defined(MOZ_WIDGET_ANDROID)
class VkImageHandle {
public:
explicit VkImageHandle(WebGPUParent* aParent,
const ffi::WGPUDeviceId aDeviceId,
ffi::WGPUVkImageHandle* aVkImageHandle)
: mParent(aParent),
mDeviceId(aDeviceId),
mVkImageHandle(aVkImageHandle) {}
const ffi::WGPUVkImageHandle* Get() { return mVkImageHandle; }
~VkImageHandle();
protected:
const WeakPtr<WebGPUParent> mParent;
const RawId mDeviceId;
ffi::WGPUVkImageHandle* mVkImageHandle;
};
class VkSemaphoreHandle {
public:
explicit VkSemaphoreHandle(WebGPUParent* aParent,
const ffi::WGPUDeviceId aDeviceId,
ffi::WGPUVkSemaphoreHandle* aVkSemaphoreHandle)
: mParent(aParent),
mDeviceId(aDeviceId),
mVkSemaphoreHandle(aVkSemaphoreHandle) {}
const ffi::WGPUVkSemaphoreHandle* Get() { return mVkSemaphoreHandle; }
~VkSemaphoreHandle();
protected:
const WeakPtr<WebGPUParent> mParent;
const RawId mDeviceId;
ffi::WGPUVkSemaphoreHandle* mVkSemaphoreHandle;
};
#endif
} // namespace webgpu
} // namespace mozilla
#endif // WEBGPU_PARENT_H_
|