diff --git a/backends/imgui_impl_wgpu.cpp b/backends/imgui_impl_wgpu.cpp index 3187be29a..e55f3080b 100644 --- a/backends/imgui_impl_wgpu.cpp +++ b/backends/imgui_impl_wgpu.cpp @@ -687,6 +687,7 @@ bool ImGui_ImplWGPU_CreateDeviceObjects() // Vertex input configuration WGPUVertexAttribute attribute_desc[] = { + #ifdef IMGUI_IMPL_WEBGPU_BACKEND_DAWN { nullptr, WGPUVertexFormat_Float32x2, (uint64_t)offsetof(ImDrawVert, pos), 0 }, { nullptr, WGPUVertexFormat_Float32x2, (uint64_t)offsetof(ImDrawVert, uv), 1 }, diff --git a/examples/example_glfw_wgpu/CMakeLists.txt b/examples/example_glfw_wgpu/CMakeLists.txt old mode 100644 new mode 100755 index 8e164e488..d85f539cd --- a/examples/example_glfw_wgpu/CMakeLists.txt +++ b/examples/example_glfw_wgpu/CMakeLists.txt @@ -62,6 +62,15 @@ else() option(TINT_BUILD_WGSL_WRITER "Build the WGSL output writer" ON) endif() +# check if WAYLAND is the current Session Type and enable DAWN_USE_WAYLAND Wayland option @compile time +# You can override this using: cmake -DDAWN_USE_WAYLAND=X (X = ON | OFF) + if(UNIX) + if ($ENV{XDG_SESSION_TYPE} MATCHES wayland) + option(DAWN_USE_WAYLAND "Enable support for Wayland surface" ON) + endif() + endif() + + add_subdirectory("${IMGUI_DAWN_DIR}" "${CMAKE_CURRENT_BINARY_DIR}/dawn" EXCLUDE_FROM_ALL) set(LIBRARIES webgpu_dawn webgpu_cpp webgpu_glfw glfw) diff --git a/examples/example_glfw_wgpu/main.cpp b/examples/example_glfw_wgpu/main.cpp old mode 100644 new mode 100755 index f510987ed..9f0242245 --- a/examples/example_glfw_wgpu/main.cpp +++ b/examples/example_glfw_wgpu/main.cpp @@ -22,44 +22,104 @@ #endif #include -#include -#include // This example can also compile and run with Emscripten! See 'Makefile.emscripten' for details. #ifdef __EMSCRIPTEN__ +#include +#include #include "../libs/emscripten/emscripten_mainloop_stub.h" #endif // Global WebGPU required states -static WGPUInstance wgpu_instance = nullptr; -static WGPUDevice wgpu_device = nullptr; -static WGPUSurface wgpu_surface = nullptr; -static WGPUTextureFormat wgpu_preferred_fmt = WGPUTextureFormat_RGBA8Unorm; -static WGPUSwapChain wgpu_swap_chain = nullptr; -static int wgpu_swap_chain_width = 1280; -static int wgpu_swap_chain_height = 720; +static WGPUInstance wgpu_instance = nullptr; +static WGPUDevice wgpu_device = nullptr; +static WGPUSurface wgpu_surface = nullptr; +static WGPUQueue wgpu_queue = nullptr; +static WGPUTextureFormat wgpu_preferred_fmt = WGPUTextureFormat_Undefined; // acquired from SurfaceCapabilities +static WGPUSurfaceConfiguration wgpu_surface_configuration {}; +static int wgpu_surface_width = 1280; +static int wgpu_surface_height = 720; // Forward declarations static bool InitWGPU(GLFWwindow* window); -static void CreateSwapChain(int width, int height); +static void ResizeSurface(int width, int height); + +#ifndef __EMSCRIPTEN__ +static void wgpu_device_lost_callback(const wgpu::Device&, wgpu::DeviceLostReason reason, wgpu::StringView message) +{ + const char* reasonName = ""; + switch (reason) + { + case wgpu::DeviceLostReason::Unknown: reasonName = "Unknown"; break; + case wgpu::DeviceLostReason::Destroyed: reasonName = "Destroyed"; break; + case wgpu::DeviceLostReason::CallbackCancelled: reasonName = "InstanceDropped"; break; + case wgpu::DeviceLostReason::FailedCreation: reasonName = "FailedCreation"; break; + default: reasonName = "UNREACHABLE"; break; + } + printf("%s device message: %s\n", reasonName, message.data); +} + +static void wgpu_error_callback(const wgpu::Device&, wgpu::ErrorType type, wgpu::StringView message) +{ + const char* errorTypeName = ""; + switch (type) + { + case wgpu::ErrorType::Validation: errorTypeName = "Validation"; break; + case wgpu::ErrorType::OutOfMemory: errorTypeName = "Out of memory"; break; + case wgpu::ErrorType::Unknown: errorTypeName = "Unknown"; break; + case wgpu::ErrorType::Internal: errorTypeName = "Internal"; break; + default: errorTypeName = "UNREACHABLE"; break; + } + printf("%s error: %s\n", errorTypeName, message.data); +} +#endif static void glfw_error_callback(int error, const char* description) { printf("GLFW Error %d: %s\n", error, description); } -static void wgpu_error_callback(WGPUErrorType error_type, const char* message, void*) +static WGPUTexture check_surface_texture_status(GLFWwindow * window) { - const char* error_type_lbl = ""; - switch (error_type) + WGPUSurfaceTexture surfaceTexture; + wgpuSurfaceGetCurrentTexture(wgpu_surface, &surfaceTexture); + + switch ( surfaceTexture.status ) { - case WGPUErrorType_Validation: error_type_lbl = "Validation"; break; - case WGPUErrorType_OutOfMemory: error_type_lbl = "Out of memory"; break; - case WGPUErrorType_Unknown: error_type_lbl = "Unknown"; break; - case WGPUErrorType_DeviceLost: error_type_lbl = "Device lost"; break; - default: error_type_lbl = "Unknown"; +#if !defined(IMGUI_IMPL_WEBGPU_BACKEND_DAWN) + case WGPUSurfaceGetCurrentTextureStatus_Success: + break; +#else + case WGPUSurfaceGetCurrentTextureStatus_SuccessOptimal: + break; + case WGPUSurfaceGetCurrentTextureStatus_SuccessSuboptimal: +#endif + case WGPUSurfaceGetCurrentTextureStatus_Timeout: + case WGPUSurfaceGetCurrentTextureStatus_Outdated: + case WGPUSurfaceGetCurrentTextureStatus_Lost: + // if the status is NOT Optimal let's try to reconfigure the surface + { +#ifndef NDEBUG + printf("Bad surface texture status: %d\n", surfaceTexture.status); +#endif + if (surfaceTexture.texture) + wgpuTextureRelease(surfaceTexture.texture); + int width, height; + glfwGetFramebufferSize(window, &width, &height); + if ( width > 0 && height > 0 ) + { + wgpu_surface_configuration.width = width; + wgpu_surface_configuration.height = height; + + wgpuSurfaceConfigure(wgpu_surface, &wgpu_surface_configuration); + } + return nullptr; + } + default: // should never be reached + assert(!"Unexpected Surface Texture status error\n"); + return nullptr; } - printf("%s error: %s\n", error_type_lbl, message); + return surfaceTexture.texture; } // Main code @@ -72,19 +132,18 @@ int main(int, char**) // Make sure GLFW does not initialize any graphics context. // This needs to be done explicitly later. glfwWindowHint(GLFW_CLIENT_API, GLFW_NO_API); - GLFWwindow* window = glfwCreateWindow(wgpu_swap_chain_width, wgpu_swap_chain_height, "Dear ImGui GLFW+WebGPU example", nullptr, nullptr); + GLFWwindow* window = glfwCreateWindow(wgpu_surface_width, wgpu_surface_height, "Dear ImGui GLFW+WebGPU example", nullptr, nullptr); if (window == nullptr) return 1; // Initialize the WebGPU environment if (!InitWGPU(window)) { - if (window) - glfwDestroyWindow(window); + glfwDestroyWindow(window); glfwTerminate(); return 1; } - CreateSwapChain(wgpu_swap_chain_width, wgpu_swap_chain_height); + glfwShowWindow(window); // Setup Dear ImGui context @@ -160,13 +219,17 @@ int main(int, char**) // React to changes in screen size int width, height; glfwGetFramebufferSize((GLFWwindow*)window, &width, &height); - if (width != wgpu_swap_chain_width || height != wgpu_swap_chain_height) + if (width != wgpu_surface_width || height != wgpu_surface_height) { ImGui_ImplWGPU_InvalidateDeviceObjects(); - CreateSwapChain(width, height); + ResizeSurface(width, height); ImGui_ImplWGPU_CreateDeviceObjects(); } + // Check surface texture status + WGPUTexture texture = check_surface_texture_status(window); + if(!texture) continue; + // Start the Dear ImGui frame ImGui_ImplWGPU_NewFrame(); ImGui_ImplGlfw_NewFrame(); @@ -212,21 +275,25 @@ int main(int, char**) // Rendering ImGui::Render(); -#ifndef __EMSCRIPTEN__ - // Tick needs to be called in Dawn to display validation errors - wgpuDeviceTick(wgpu_device); -#endif + WGPUTextureViewDescriptor viewDescriptor {}; + viewDescriptor.format = wgpu_preferred_fmt; + viewDescriptor.dimension = WGPUTextureViewDimension_2D ; + viewDescriptor.mipLevelCount = WGPU_MIP_LEVEL_COUNT_UNDEFINED; + viewDescriptor.arrayLayerCount = WGPU_ARRAY_LAYER_COUNT_UNDEFINED; + viewDescriptor.aspect = WGPUTextureAspect_All; - WGPURenderPassColorAttachment color_attachments = {}; + WGPUTextureView textureView = wgpuTextureCreateView(texture, &viewDescriptor); + + WGPURenderPassColorAttachment color_attachments {}; color_attachments.depthSlice = WGPU_DEPTH_SLICE_UNDEFINED; - color_attachments.loadOp = WGPULoadOp_Clear; - color_attachments.storeOp = WGPUStoreOp_Store; + color_attachments.loadOp = WGPULoadOp_Clear; + color_attachments.storeOp = WGPUStoreOp_Store; color_attachments.clearValue = { clear_color.x * clear_color.w, clear_color.y * clear_color.w, clear_color.z * clear_color.w, clear_color.w }; - color_attachments.view = wgpuSwapChainGetCurrentTextureView(wgpu_swap_chain); + color_attachments.view = textureView; - WGPURenderPassDescriptor render_pass_desc = {}; - render_pass_desc.colorAttachmentCount = 1; - render_pass_desc.colorAttachments = &color_attachments; + WGPURenderPassDescriptor render_pass_desc {}; + render_pass_desc.colorAttachmentCount = 1; + render_pass_desc.colorAttachments = &color_attachments; render_pass_desc.depthStencilAttachment = nullptr; WGPUCommandEncoderDescriptor enc_desc = {}; @@ -236,16 +303,16 @@ int main(int, char**) ImGui_ImplWGPU_RenderDrawData(ImGui::GetDrawData(), pass); wgpuRenderPassEncoderEnd(pass); - WGPUCommandBufferDescriptor cmd_buffer_desc = {}; + WGPUCommandBufferDescriptor cmd_buffer_desc {}; WGPUCommandBuffer cmd_buffer = wgpuCommandEncoderFinish(encoder, &cmd_buffer_desc); - WGPUQueue queue = wgpuDeviceGetQueue(wgpu_device); - wgpuQueueSubmit(queue, 1, &cmd_buffer); + wgpuQueueSubmit(wgpu_queue, 1, &cmd_buffer); #ifndef __EMSCRIPTEN__ - wgpuSwapChainPresent(wgpu_swap_chain); + wgpuSurfacePresent(wgpu_surface); + // Tick needs to be called in Dawn to display validation errors + wgpuDeviceTick(wgpu_device); #endif - - wgpuTextureViewRelease(color_attachments.view); + wgpuTextureViewRelease(textureView); wgpuRenderPassEncoderRelease(pass); wgpuCommandEncoderRelease(encoder); wgpuCommandBufferRelease(cmd_buffer); @@ -259,58 +326,32 @@ int main(int, char**) ImGui_ImplGlfw_Shutdown(); ImGui::DestroyContext(); + wgpuSurfaceUnconfigure(wgpu_surface); + wgpuSurfaceRelease(wgpu_surface); + wgpuQueueRelease(wgpu_queue); + wgpuDeviceRelease(wgpu_device); + wgpuInstanceRelease(wgpu_instance); + glfwDestroyWindow(window); glfwTerminate(); return 0; } -#ifndef __EMSCRIPTEN__ -static WGPUAdapter RequestAdapter(WGPUInstance instance) -{ - auto onAdapterRequestEnded = [](WGPURequestAdapterStatus status, WGPUAdapter adapter, const char* message, void* pUserData) - { - if (status == WGPURequestAdapterStatus_Success) - *(WGPUAdapter*)(pUserData) = adapter; - else - printf("Could not get WebGPU adapter: %s\n", message); -}; - WGPUAdapter adapter; - wgpuInstanceRequestAdapter(instance, nullptr, onAdapterRequestEnded, (void*)&adapter); - return adapter; -} - -static WGPUDevice RequestDevice(WGPUAdapter& adapter) -{ - auto onDeviceRequestEnded = [](WGPURequestDeviceStatus status, WGPUDevice device, const char* message, void* pUserData) - { - if (status == WGPURequestDeviceStatus_Success) - *(WGPUDevice*)(pUserData) = device; - else - printf("Could not get WebGPU device: %s\n", message); - }; - WGPUDevice device; - wgpuAdapterRequestDevice(adapter, nullptr, onDeviceRequestEnded, (void*)&device); - return device; -} -#endif - static bool InitWGPU(GLFWwindow* window) { - wgpu::Instance instance = wgpuCreateInstance(nullptr); + wgpu_surface_configuration.presentMode = WGPUPresentMode_Fifo; + wgpu_surface_configuration.alphaMode = WGPUCompositeAlphaMode_Auto; + wgpu_surface_configuration.usage = WGPUTextureUsage_RenderAttachment; + wgpu_surface_configuration.width = wgpu_surface_width; + wgpu_surface_configuration.height = wgpu_surface_height; #ifdef __EMSCRIPTEN__ + wgpu::Instance instance = wgpuCreateInstance(nullptr); wgpu_device = emscripten_webgpu_get_device(); if (!wgpu_device) return false; -#else - WGPUAdapter adapter = RequestAdapter(instance.Get()); - if (!adapter) - return false; - wgpu_device = RequestDevice(adapter); -#endif -#ifdef __EMSCRIPTEN__ wgpu::SurfaceDescriptorFromCanvasHTMLSelector html_surface_desc = {}; html_surface_desc.selector = "#canvas"; wgpu::SurfaceDescriptor surface_desc = {}; @@ -319,32 +360,73 @@ static bool InitWGPU(GLFWwindow* window) wgpu::Adapter adapter = {}; wgpu_preferred_fmt = (WGPUTextureFormat)surface.GetPreferredFormat(adapter); + + wgpu_surface_configuration.device = wgpu_device; + wgpu_surface_configuration.format = wgpu_preferred_fmt; #else + + wgpu::InstanceDescriptor instanceDescriptor {}; + instanceDescriptor.capabilities.timedWaitAnyEnable = true; + wgpu::Instance instance = wgpu::CreateInstance(&instanceDescriptor); + + static wgpu::Adapter localAdapter; + wgpu::RequestAdapterOptions adapterOptions {}; + + auto onRequestAdapter = [](wgpu::RequestAdapterStatus status, wgpu::Adapter adapter, wgpu::StringView message) + { + if (status != wgpu::RequestAdapterStatus::Success) + { + printf("Failed to get an adapter: %s\n", message.data); + return; + } + localAdapter = std::move(adapter); + }; + + // Synchronously (wait until) acquire Adapter + auto waitedAdapterFunc { instance.RequestAdapter(&adapterOptions, wgpu::CallbackMode::WaitAnyOnly, onRequestAdapter) }; + instance.WaitAny(waitedAdapterFunc, UINT64_MAX); + assert(localAdapter != nullptr); + +#ifndef NDEBUG + wgpu::AdapterInfo info; + localAdapter.GetInfo(&info); + printf("Using adapter: \" %s \"\n", info.device.data); +#endif + + // Set device callback functions + wgpu::DeviceDescriptor deviceDesc {}; + deviceDesc.SetDeviceLostCallback(wgpu::CallbackMode::AllowSpontaneous, wgpu_device_lost_callback); + deviceDesc.SetUncapturedErrorCallback(wgpu_error_callback); + + // get device Synchronously + wgpu_device = localAdapter.CreateDevice(&deviceDesc).MoveToCHandle(); + assert(wgpu_device != nullptr); + wgpu::Surface surface = wgpu::glfw::CreateSurfaceForWindow(instance, window); if (!surface) return false; - wgpu_preferred_fmt = WGPUTextureFormat_BGRA8Unorm; + + // Configure the surface. + wgpu::SurfaceCapabilities capabilities; + surface.GetCapabilities(localAdapter, &capabilities); + wgpu_preferred_fmt = (WGPUTextureFormat) capabilities.formats[0]; + + wgpu_surface_configuration.device = wgpu_device; + wgpu_surface_configuration.format = wgpu_preferred_fmt; + surface.Configure((const wgpu::SurfaceConfiguration *) &wgpu_surface_configuration); #endif wgpu_instance = instance.MoveToCHandle(); - wgpu_surface = surface.MoveToCHandle(); - - wgpuDeviceSetUncapturedErrorCallback(wgpu_device, wgpu_error_callback, nullptr); + wgpu_surface = surface.MoveToCHandle(); + wgpu_queue = wgpuDeviceGetQueue(wgpu_device); return true; } -static void CreateSwapChain(int width, int height) +void ResizeSurface(int width, int height) { - if (wgpu_swap_chain) - wgpuSwapChainRelease(wgpu_swap_chain); - wgpu_swap_chain_width = width; - wgpu_swap_chain_height = height; - WGPUSwapChainDescriptor swap_chain_desc = {}; - swap_chain_desc.usage = WGPUTextureUsage_RenderAttachment; - swap_chain_desc.format = wgpu_preferred_fmt; - swap_chain_desc.width = width; - swap_chain_desc.height = height; - swap_chain_desc.presentMode = WGPUPresentMode_Fifo; - wgpu_swap_chain = wgpuDeviceCreateSwapChain(wgpu_device, wgpu_surface, &swap_chain_desc); + wgpu_surface_configuration.width = wgpu_surface_width = width; + wgpu_surface_configuration.height = wgpu_surface_height = height; + + wgpuSurfaceConfigure(wgpu_surface, (WGPUSurfaceConfiguration *) &wgpu_surface_configuration); } diff --git a/examples/example_glfw_wgpu/web/index.html b/examples/example_glfw_wgpu/web/index.html old mode 100644 new mode 100755 index a2a91c4a7..8257d78d3 --- a/examples/example_glfw_wgpu/web/index.html +++ b/examples/example_glfw_wgpu/web/index.html @@ -5,7 +5,7 @@ Dear ImGui Emscripten+GLFW+WebGPU example + + + + + +