Skip to content

Commit 015186a

Browse files
committed
Merge branch 'master' into docking
# Conflicts: # backends/imgui_impl_dx12.cpp # backends/imgui_impl_vulkan.cpp
2 parents b9badb5 + 0f33d71 commit 015186a

File tree

18 files changed

+124
-106
lines changed

18 files changed

+124
-106
lines changed

backends/imgui_impl_dx10.cpp

+1-1
Original file line numberDiff line numberDiff line change
@@ -49,7 +49,7 @@
4949
#pragma comment(lib, "d3dcompiler") // Automatically link with d3dcompiler.lib as we are using D3DCompile() below.
5050
#endif
5151

52-
// DirectX data
52+
// DirectX10 data
5353
struct ImGui_ImplDX10_Data
5454
{
5555
ID3D10Device* pd3dDevice;

backends/imgui_impl_dx12.cpp

+22-27
Original file line numberDiff line numberDiff line change
@@ -22,6 +22,7 @@
2222
// CHANGELOG
2323
// (minor and older changes stripped away, please see git history for details)
2424
// 2025-XX-XX: Platform: Added support for multiple windows via the ImGuiPlatformIO interface.
25+
// 2025-01-15: DirectX12: Texture upload use the command queue provided in ImGui_ImplDX12_InitInfo instead of creating its own.
2526
// 2024-12-09: DirectX12: Let user specifies the DepthStencilView format by setting ImGui_ImplDX12_InitInfo::DSVFormat.
2627
// 2024-11-15: DirectX12: *BREAKING CHANGE* Changed ImGui_ImplDX12_Init() signature to take a ImGui_ImplDX12_InitInfo struct. Legacy ImGui_ImplDX12_Init() signature is still supported (will obsolete).
2728
// 2024-11-15: DirectX12: *BREAKING CHANGE* User is now required to pass function pointers to allocate/free SRV Descriptors. We provide convenience legacy fields to pass a single descriptor, matching the old API, but upcoming features will want multiple.
@@ -258,6 +259,7 @@ void ImGui_ImplDX12_RenderDrawData(ImDrawData* draw_data, ID3D12GraphicsCommandL
258259
if (draw_data->DisplaySize.x <= 0.0f || draw_data->DisplaySize.y <= 0.0f)
259260
return;
260261

262+
// FIXME: We are assuming that this only gets called once per frame!
261263
ImGui_ImplDX12_Data* bd = ImGui_ImplDX12_GetBackendData();
262264
ImGui_ImplDX12_ViewportData* vd = (ImGui_ImplDX12_ViewportData*)draw_data->OwnerViewport->RendererUserData;
263265
vd->FrameIndex++;
@@ -429,11 +431,11 @@ static void ImGui_ImplDX12_CreateFontsTexture()
429431
bd->pd3dDevice->CreateCommittedResource(&props, D3D12_HEAP_FLAG_NONE, &desc,
430432
D3D12_RESOURCE_STATE_COPY_DEST, nullptr, IID_PPV_ARGS(&pTexture));
431433

432-
UINT uploadPitch = (width * 4 + D3D12_TEXTURE_DATA_PITCH_ALIGNMENT - 1u) & ~(D3D12_TEXTURE_DATA_PITCH_ALIGNMENT - 1u);
433-
UINT uploadSize = height * uploadPitch;
434+
UINT upload_pitch = (width * 4 + D3D12_TEXTURE_DATA_PITCH_ALIGNMENT - 1u) & ~(D3D12_TEXTURE_DATA_PITCH_ALIGNMENT - 1u);
435+
UINT upload_size = height * upload_pitch;
434436
desc.Dimension = D3D12_RESOURCE_DIMENSION_BUFFER;
435437
desc.Alignment = 0;
436-
desc.Width = uploadSize;
438+
desc.Width = upload_size;
437439
desc.Height = 1;
438440
desc.DepthOrArraySize = 1;
439441
desc.MipLevels = 1;
@@ -453,26 +455,28 @@ static void ImGui_ImplDX12_CreateFontsTexture()
453455
IM_ASSERT(SUCCEEDED(hr));
454456

455457
void* mapped = nullptr;
456-
D3D12_RANGE range = { 0, uploadSize };
458+
D3D12_RANGE range = { 0, upload_size };
457459
hr = uploadBuffer->Map(0, &range, &mapped);
458460
IM_ASSERT(SUCCEEDED(hr));
459461
for (int y = 0; y < height; y++)
460-
memcpy((void*) ((uintptr_t) mapped + y * uploadPitch), pixels + y * width * 4, width * 4);
462+
memcpy((void*) ((uintptr_t) mapped + y * upload_pitch), pixels + y * width * 4, width * 4);
461463
uploadBuffer->Unmap(0, &range);
462464

463465
D3D12_TEXTURE_COPY_LOCATION srcLocation = {};
464-
srcLocation.pResource = uploadBuffer;
465-
srcLocation.Type = D3D12_TEXTURE_COPY_TYPE_PLACED_FOOTPRINT;
466-
srcLocation.PlacedFootprint.Footprint.Format = DXGI_FORMAT_R8G8B8A8_UNORM;
467-
srcLocation.PlacedFootprint.Footprint.Width = width;
468-
srcLocation.PlacedFootprint.Footprint.Height = height;
469-
srcLocation.PlacedFootprint.Footprint.Depth = 1;
470-
srcLocation.PlacedFootprint.Footprint.RowPitch = uploadPitch;
471-
472466
D3D12_TEXTURE_COPY_LOCATION dstLocation = {};
473-
dstLocation.pResource = pTexture;
474-
dstLocation.Type = D3D12_TEXTURE_COPY_TYPE_SUBRESOURCE_INDEX;
475-
dstLocation.SubresourceIndex = 0;
467+
{
468+
srcLocation.pResource = uploadBuffer;
469+
srcLocation.Type = D3D12_TEXTURE_COPY_TYPE_PLACED_FOOTPRINT;
470+
srcLocation.PlacedFootprint.Footprint.Format = DXGI_FORMAT_R8G8B8A8_UNORM;
471+
srcLocation.PlacedFootprint.Footprint.Width = width;
472+
srcLocation.PlacedFootprint.Footprint.Height = height;
473+
srcLocation.PlacedFootprint.Footprint.Depth = 1;
474+
srcLocation.PlacedFootprint.Footprint.RowPitch = upload_pitch;
475+
476+
dstLocation.pResource = pTexture;
477+
dstLocation.Type = D3D12_TEXTURE_COPY_TYPE_SUBRESOURCE_INDEX;
478+
dstLocation.SubresourceIndex = 0;
479+
}
476480

477481
D3D12_RESOURCE_BARRIER barrier = {};
478482
barrier.Type = D3D12_RESOURCE_BARRIER_TYPE_TRANSITION;
@@ -489,15 +493,6 @@ static void ImGui_ImplDX12_CreateFontsTexture()
489493
HANDLE event = ::CreateEvent(0, 0, 0, 0);
490494
IM_ASSERT(event != nullptr);
491495

492-
D3D12_COMMAND_QUEUE_DESC queueDesc = {};
493-
queueDesc.Type = D3D12_COMMAND_LIST_TYPE_DIRECT;
494-
queueDesc.Flags = D3D12_COMMAND_QUEUE_FLAG_NONE;
495-
queueDesc.NodeMask = 1;
496-
497-
ID3D12CommandQueue* cmdQueue = nullptr;
498-
hr = bd->pd3dDevice->CreateCommandQueue(&queueDesc, IID_PPV_ARGS(&cmdQueue));
499-
IM_ASSERT(SUCCEEDED(hr));
500-
501496
ID3D12CommandAllocator* cmdAlloc = nullptr;
502497
hr = bd->pd3dDevice->CreateCommandAllocator(D3D12_COMMAND_LIST_TYPE_DIRECT, IID_PPV_ARGS(&cmdAlloc));
503498
IM_ASSERT(SUCCEEDED(hr));
@@ -512,6 +507,7 @@ static void ImGui_ImplDX12_CreateFontsTexture()
512507
hr = cmdList->Close();
513508
IM_ASSERT(SUCCEEDED(hr));
514509

510+
ID3D12CommandQueue* cmdQueue = bd->InitInfo.CommandQueue;
515511
cmdQueue->ExecuteCommandLists(1, (ID3D12CommandList* const*)&cmdList);
516512
hr = cmdQueue->Signal(fence, 1);
517513
IM_ASSERT(SUCCEEDED(hr));
@@ -521,7 +517,6 @@ static void ImGui_ImplDX12_CreateFontsTexture()
521517

522518
cmdList->Release();
523519
cmdAlloc->Release();
524-
cmdQueue->Release();
525520
::CloseHandle(event);
526521
fence->Release();
527522
uploadBuffer->Release();
@@ -791,11 +786,11 @@ void ImGui_ImplDX12_InvalidateDeviceObjects()
791786
if (!bd || !bd->pd3dDevice)
792787
return;
793788

794-
ImGuiIO& io = ImGui::GetIO();
795789
SafeRelease(bd->pRootSignature);
796790
SafeRelease(bd->pPipelineState);
797791

798792
// Free SRV descriptor used by texture
793+
ImGuiIO& io = ImGui::GetIO();
799794
ImGui_ImplDX12_Texture* font_tex = &bd->FontTexture;
800795
bd->InitInfo.SrvDescriptorFreeFn(&bd->InitInfo, font_tex->hFontSrvCpuDescHandle, font_tex->hFontSrvGpuDescHandle);
801796
SafeRelease(font_tex->pTextureResource);

backends/imgui_impl_opengl2.cpp

+11-1
Original file line numberDiff line numberDiff line change
@@ -71,6 +71,16 @@
7171
#include <GL/gl.h>
7272
#endif
7373

74+
// [Debugging]
75+
//#define IMGUI_IMPL_OPENGL_DEBUG
76+
#ifdef IMGUI_IMPL_OPENGL_DEBUG
77+
#include <stdio.h>
78+
#define GL_CALL(_CALL) do { _CALL; GLenum gl_err = glGetError(); if (gl_err != 0) fprintf(stderr, "GL error 0x%x returned from '%s'.\n", gl_err, #_CALL); } while (0) // Call with error check
79+
#else
80+
#define GL_CALL(_CALL) _CALL // Call without error check
81+
#endif
82+
83+
// OpenGL data
7484
struct ImGui_ImplOpenGL2_Data
7585
{
7686
GLuint FontTexture;
@@ -166,7 +176,7 @@ static void ImGui_ImplOpenGL2_SetupRenderState(ImDrawData* draw_data, int fb_wid
166176

167177
// Setup viewport, orthographic projection matrix
168178
// Our visible imgui space lies from draw_data->DisplayPos (top left) to draw_data->DisplayPos+data_data->DisplaySize (bottom right). DisplayPos is (0,0) for single viewport apps.
169-
glViewport(0, 0, (GLsizei)fb_width, (GLsizei)fb_height);
179+
GL_CALL(glViewport(0, 0, (GLsizei)fb_width, (GLsizei)fb_height));
170180
glMatrixMode(GL_PROJECTION);
171181
glPushMatrix();
172182
glLoadIdentity();

backends/imgui_impl_vulkan.cpp

+13-19
Original file line numberDiff line numberDiff line change
@@ -226,7 +226,7 @@ struct ImGui_ImplVulkan_WindowRenderBuffers
226226
{
227227
uint32_t Index;
228228
uint32_t Count;
229-
ImGui_ImplVulkan_FrameRenderBuffers* FrameRenderBuffers;
229+
ImVector<ImGui_ImplVulkan_FrameRenderBuffers> FrameRenderBuffers;
230230
};
231231

232232
struct ImGui_ImplVulkan_Texture
@@ -531,12 +531,12 @@ void ImGui_ImplVulkan_RenderDrawData(ImDrawData* draw_data, VkCommandBuffer comm
531531
ImGui_ImplVulkan_ViewportData* viewport_renderer_data = (ImGui_ImplVulkan_ViewportData*)draw_data->OwnerViewport->RendererUserData;
532532
IM_ASSERT(viewport_renderer_data != nullptr);
533533
ImGui_ImplVulkan_WindowRenderBuffers* wrb = &viewport_renderer_data->RenderBuffers;
534-
if (wrb->FrameRenderBuffers == nullptr)
534+
if (wrb->FrameRenderBuffers.Size == 0)
535535
{
536536
wrb->Index = 0;
537537
wrb->Count = v->ImageCount;
538-
wrb->FrameRenderBuffers = (ImGui_ImplVulkan_FrameRenderBuffers*)IM_ALLOC(sizeof(ImGui_ImplVulkan_FrameRenderBuffers) * wrb->Count);
539-
memset((void*)wrb->FrameRenderBuffers, 0, sizeof(ImGui_ImplVulkan_FrameRenderBuffers) * wrb->Count);
538+
wrb->FrameRenderBuffers.resize(wrb->Count);
539+
memset((void*)wrb->FrameRenderBuffers.Data, 0, wrb->FrameRenderBuffers.size_in_bytes());
540540
}
541541
IM_ASSERT(wrb->Count == v->ImageCount);
542542
wrb->Index = (wrb->Index + 1) % wrb->Count;
@@ -1301,8 +1301,7 @@ void ImGui_ImplVulkan_DestroyWindowRenderBuffers(VkDevice device, ImGui_ImplVulk
13011301
{
13021302
for (uint32_t n = 0; n < buffers->Count; n++)
13031303
ImGui_ImplVulkan_DestroyFrameRenderBuffers(device, &buffers->FrameRenderBuffers[n], allocator);
1304-
IM_FREE(buffers->FrameRenderBuffers);
1305-
buffers->FrameRenderBuffers = nullptr;
1304+
buffers->FrameRenderBuffers.clear();
13061305
buffers->Index = 0;
13071306
buffers->Count = 0;
13081307
}
@@ -1511,10 +1510,8 @@ void ImGui_ImplVulkanH_CreateWindowSwapChain(VkPhysicalDevice physical_device, V
15111510
ImGui_ImplVulkanH_DestroyFrame(device, &wd->Frames[i], allocator);
15121511
for (uint32_t i = 0; i < wd->SemaphoreCount; i++)
15131512
ImGui_ImplVulkanH_DestroyFrameSemaphores(device, &wd->FrameSemaphores[i], allocator);
1514-
IM_FREE(wd->Frames);
1515-
IM_FREE(wd->FrameSemaphores);
1516-
wd->Frames = nullptr;
1517-
wd->FrameSemaphores = nullptr;
1513+
wd->Frames.clear();
1514+
wd->FrameSemaphores.clear();
15181515
wd->ImageCount = 0;
15191516
if (wd->RenderPass)
15201517
vkDestroyRenderPass(device, wd->RenderPass, allocator);
@@ -1567,12 +1564,11 @@ void ImGui_ImplVulkanH_CreateWindowSwapChain(VkPhysicalDevice physical_device, V
15671564
err = vkGetSwapchainImagesKHR(device, wd->Swapchain, &wd->ImageCount, backbuffers);
15681565
check_vk_result(err);
15691566

1570-
IM_ASSERT(wd->Frames == nullptr && wd->FrameSemaphores == nullptr);
15711567
wd->SemaphoreCount = wd->ImageCount + 1;
1572-
wd->Frames = (ImGui_ImplVulkanH_Frame*)IM_ALLOC(sizeof(ImGui_ImplVulkanH_Frame) * wd->ImageCount);
1573-
wd->FrameSemaphores = (ImGui_ImplVulkanH_FrameSemaphores*)IM_ALLOC(sizeof(ImGui_ImplVulkanH_FrameSemaphores) * wd->SemaphoreCount);
1574-
memset((void*)wd->Frames, 0, sizeof(wd->Frames[0]) * wd->ImageCount);
1575-
memset((void*)wd->FrameSemaphores, 0, sizeof(wd->FrameSemaphores[0]) * wd->SemaphoreCount);
1568+
wd->Frames.resize(wd->ImageCount);
1569+
wd->FrameSemaphores.resize(wd->SemaphoreCount);
1570+
memset(wd->Frames.Data, 0, wd->Frames.size_in_bytes());
1571+
memset(wd->FrameSemaphores.Data, 0, wd->FrameSemaphores.size_in_bytes());
15761572
for (uint32_t i = 0; i < wd->ImageCount; i++)
15771573
wd->Frames[i].Backbuffer = backbuffers[i];
15781574
}
@@ -1683,10 +1679,8 @@ void ImGui_ImplVulkanH_DestroyWindow(VkInstance instance, VkDevice device, ImGui
16831679
ImGui_ImplVulkanH_DestroyFrame(device, &wd->Frames[i], allocator);
16841680
for (uint32_t i = 0; i < wd->SemaphoreCount; i++)
16851681
ImGui_ImplVulkanH_DestroyFrameSemaphores(device, &wd->FrameSemaphores[i], allocator);
1686-
IM_FREE(wd->Frames);
1687-
IM_FREE(wd->FrameSemaphores);
1688-
wd->Frames = nullptr;
1689-
wd->FrameSemaphores = nullptr;
1682+
wd->Frames.clear();
1683+
wd->FrameSemaphores.clear();
16901684
vkDestroyRenderPass(device, wd->RenderPass, allocator);
16911685
vkDestroySwapchainKHR(device, wd->Swapchain, allocator);
16921686
vkDestroySurfaceKHR(instance, wd->Surface, allocator);

backends/imgui_impl_vulkan.h

+2-2
Original file line numberDiff line numberDiff line change
@@ -207,8 +207,8 @@ struct ImGui_ImplVulkanH_Window
207207
uint32_t ImageCount; // Number of simultaneous in-flight frames (returned by vkGetSwapchainImagesKHR, usually derived from min_image_count)
208208
uint32_t SemaphoreCount; // Number of simultaneous in-flight frames + 1, to be able to use it in vkAcquireNextImageKHR
209209
uint32_t SemaphoreIndex; // Current set of swapchain wait semaphores we're using (needs to be distinct from per frame data)
210-
ImGui_ImplVulkanH_Frame* Frames;
211-
ImGui_ImplVulkanH_FrameSemaphores* FrameSemaphores;
210+
ImVector<ImGui_ImplVulkanH_Frame> Frames;
211+
ImVector<ImGui_ImplVulkanH_FrameSemaphores> FrameSemaphores;
212212

213213
ImGui_ImplVulkanH_Window()
214214
{

docs/CHANGELOG.txt

+17
Original file line numberDiff line numberDiff line change
@@ -35,6 +35,23 @@ HOW TO UPDATE?
3535
and API updates have been a little more frequent lately. They are documented below and in imgui.cpp and should not affect all users.
3636
- Please report any issue!
3737

38+
-----------------------------------------------------------------------
39+
VERSION 1.91.8 WIP (In Progress)
40+
-----------------------------------------------------------------------
41+
42+
Breaking changes:
43+
44+
Other changes:
45+
46+
- ImDrawList: texture baked storage for thick line reduced from ~64x64 to ~32x32. (#3245)
47+
- Examples: DirectX12: Reduced number of frame in flight from 3 to 2 in
48+
provided example, to reduce latency.
49+
- Examples: Vulkan: better handle VK_SUBOPTIMAL_KHR being returned by
50+
vkAcquireNextImageKHR() or vkQueuePresentKHR(). (#7825, #7831) [@NostraMagister]
51+
- Backends: DirectX12: Texture upload use the command queue provided in
52+
ImGui_ImplDX12_InitInfo instead of creating its own.
53+
54+
3855
-----------------------------------------------------------------------
3956
VERSION 1.91.7 (Released 2025-01-14)
4057
-----------------------------------------------------------------------

examples/example_glfw_vulkan/main.cpp

+8-10
Original file line numberDiff line numberDiff line change
@@ -61,7 +61,7 @@ static void glfw_error_callback(int error, const char* description)
6161
}
6262
static void check_vk_result(VkResult err)
6363
{
64-
if (err == 0)
64+
if (err == VK_SUCCESS)
6565
return;
6666
fprintf(stderr, "[vulkan] Error: VkResult = %d\n", err);
6767
if (err < 0)
@@ -263,17 +263,15 @@ static void CleanupVulkanWindow()
263263

264264
static void FrameRender(ImGui_ImplVulkanH_Window* wd, ImDrawData* draw_data)
265265
{
266-
VkResult err;
267-
268266
VkSemaphore image_acquired_semaphore = wd->FrameSemaphores[wd->SemaphoreIndex].ImageAcquiredSemaphore;
269267
VkSemaphore render_complete_semaphore = wd->FrameSemaphores[wd->SemaphoreIndex].RenderCompleteSemaphore;
270-
err = vkAcquireNextImageKHR(g_Device, wd->Swapchain, UINT64_MAX, image_acquired_semaphore, VK_NULL_HANDLE, &wd->FrameIndex);
268+
VkResult err = vkAcquireNextImageKHR(g_Device, wd->Swapchain, UINT64_MAX, image_acquired_semaphore, VK_NULL_HANDLE, &wd->FrameIndex);
271269
if (err == VK_ERROR_OUT_OF_DATE_KHR || err == VK_SUBOPTIMAL_KHR)
272-
{
273270
g_SwapChainRebuild = true;
271+
if (err == VK_ERROR_OUT_OF_DATE_KHR)
274272
return;
275-
}
276-
check_vk_result(err);
273+
if (err != VK_SUBOPTIMAL_KHR)
274+
check_vk_result(err);
277275

278276
ImGui_ImplVulkanH_Frame* fd = &wd->Frames[wd->FrameIndex];
279277
{
@@ -342,11 +340,11 @@ static void FramePresent(ImGui_ImplVulkanH_Window* wd)
342340
info.pImageIndices = &wd->FrameIndex;
343341
VkResult err = vkQueuePresentKHR(g_Queue, &info);
344342
if (err == VK_ERROR_OUT_OF_DATE_KHR || err == VK_SUBOPTIMAL_KHR)
345-
{
346343
g_SwapChainRebuild = true;
344+
if (err == VK_ERROR_OUT_OF_DATE_KHR)
347345
return;
348-
}
349-
check_vk_result(err);
346+
if (err != VK_SUBOPTIMAL_KHR)
347+
check_vk_result(err);
350348
wd->SemaphoreIndex = (wd->SemaphoreIndex + 1) % wd->SemaphoreCount; // Now we can use the next set of semaphores
351349
}
352350

examples/example_sdl2_vulkan/main.cpp

+8-10
Original file line numberDiff line numberDiff line change
@@ -49,7 +49,7 @@ static bool g_SwapChainRebuild = false;
4949

5050
static void check_vk_result(VkResult err)
5151
{
52-
if (err == 0)
52+
if (err == VK_SUCCESS)
5353
return;
5454
fprintf(stderr, "[vulkan] Error: VkResult = %d\n", err);
5555
if (err < 0)
@@ -251,17 +251,15 @@ static void CleanupVulkanWindow()
251251

252252
static void FrameRender(ImGui_ImplVulkanH_Window* wd, ImDrawData* draw_data)
253253
{
254-
VkResult err;
255-
256254
VkSemaphore image_acquired_semaphore = wd->FrameSemaphores[wd->SemaphoreIndex].ImageAcquiredSemaphore;
257255
VkSemaphore render_complete_semaphore = wd->FrameSemaphores[wd->SemaphoreIndex].RenderCompleteSemaphore;
258-
err = vkAcquireNextImageKHR(g_Device, wd->Swapchain, UINT64_MAX, image_acquired_semaphore, VK_NULL_HANDLE, &wd->FrameIndex);
256+
VkResult err = vkAcquireNextImageKHR(g_Device, wd->Swapchain, UINT64_MAX, image_acquired_semaphore, VK_NULL_HANDLE, &wd->FrameIndex);
259257
if (err == VK_ERROR_OUT_OF_DATE_KHR || err == VK_SUBOPTIMAL_KHR)
260-
{
261258
g_SwapChainRebuild = true;
259+
if (err == VK_ERROR_OUT_OF_DATE_KHR)
262260
return;
263-
}
264-
check_vk_result(err);
261+
if (err != VK_SUBOPTIMAL_KHR)
262+
check_vk_result(err);
265263

266264
ImGui_ImplVulkanH_Frame* fd = &wd->Frames[wd->FrameIndex];
267265
{
@@ -330,11 +328,11 @@ static void FramePresent(ImGui_ImplVulkanH_Window* wd)
330328
info.pImageIndices = &wd->FrameIndex;
331329
VkResult err = vkQueuePresentKHR(g_Queue, &info);
332330
if (err == VK_ERROR_OUT_OF_DATE_KHR || err == VK_SUBOPTIMAL_KHR)
333-
{
334331
g_SwapChainRebuild = true;
332+
if (err == VK_ERROR_OUT_OF_DATE_KHR)
335333
return;
336-
}
337-
check_vk_result(err);
334+
if (err != VK_SUBOPTIMAL_KHR)
335+
check_vk_result(err);
338336
wd->SemaphoreIndex = (wd->SemaphoreIndex + 1) % wd->SemaphoreCount; // Now we can use the next set of semaphores
339337
}
340338

0 commit comments

Comments
 (0)