-
-
Notifications
You must be signed in to change notification settings - Fork 10.3k
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Looking for feedback/suggestions on changing ImTextureID from void* to u64 #1641
Comments
A D3D12_GPU_DESCRIPTOR_HANDLE is just a wrapper around a 64-bit GPU virtual address. I don't see the advantage of using a D3D12_GPU_DESCRIPTOR_HANDLE* vs a D3D12_GPU_DESCRIPTOR_HANDLE directly - can you clarify? I think I'm missing something. |
The advantage is that it would always fit in a pointer and therefore work in 32-bit build. Currently we can’t store them in a void* on 32-bit builds. this is what this thread describe and is trying to solve. If your handles are stored in a persistent way on the heap it doesn’t really matter if you pass them by address or by value. |
I'll preface this by mentioning that I'm not using C++ directly -- I'm downstream of the C bindings, so some of the above doesn't have much of an impact on me (like whether pointer types are convertible, etc.). If the type changes from pointer-sized to 8-byte-sized, then it will just constitute another breaking change that isn't terribly difficult to deal with. I think it might help to mention that in many graphics API's, especially modern ones, you do not bind textures directly. So I don't expect that many people are passing an actual "texture ID" in here. OpenGL and Metal are probably the only common API's where a texture is directly bindable. Even D3D11 (not a "modern" API) uses Shader Resource Views, which are a distinct resource. If you actually pass an API's texture object into these functions, it implies that you are doing a lot of "extra stuff" behind the scenes to make those into actually-bindable resources on-the-fly. In my library/engine, I just treat the ImTextureID as an opaque identifier, used as an index into a managed resource cache. You can register a texture (as in, an actual Texture, not a view/descriptor), and it internally creates the actual "bindable" resource that is needed for the API (I support D3D11, Metal, OpenGL, Vulkan). The texture IDs are assigned by literally just bumping an index. When you pass that index into Anyways, I hope that my perspective helps. Like I said, I'm not using C++ directly, so most of this would be inconsequential for me. |
Pointer to Vulcan Possible, not yet mentioned alternatives: (F) Explicit Texture API IMGUI_API ImTextureID ImGui_CreateTexture32(const void* data, int width, int height);
IMGUI_API ImTextureID ImGui_CreateTextureFromDescriptor(...); // API dependent
IMGUI_API void ImGui_DestroyTexture(ImTextureID texture);
IMGUI_API int ImGui_GetTextureWidth(ImTextureID texture);
IMGUI_API int ImGui_GetTextureHeight(ImTextureID texture); (G) State ImTextureID type in imconfig.h |
Ahem, yes, 32-bit apps still exist. I vote for option D. |
in imgui.h change
to
That way the end user could (for example) |
@thedmd In issue #914, (F) is roughly the method I went with. The reason is that I wanted to leave the ImGui Vulkan interface mostly alone. If I have to externally provide the descriptor_set then I have to also supply things I wouldn't really want to care about, like the sampler. (F) allows ImGui to hide those details under the covers. The difference between (F) and what I did, is that I have a variant taking a VkImage:
|
I am using imGUI (I LOVE IT!) with my own engine using Vulkan. And i'm strongly against changing the imGUI API sending actual API pointers through imGUI. ImGUI shouldn't care about back-end engine resources at all. It should only pass on identifiers like a GUID or some simple INT index to identify a certain resource. And the renderer will take care of that (because you have in your engine a resource manager that can translate this). It's quite dangerous for a engine framework to keep API resources around in a immediate UI interface like ImGUI with e.g an Vulkan API that does quite the opposite e.g baking resources before you even draw something. I am not sure if this makes any sense but keeping imGUI as is.. is your best option and leave it simple.. don't mix it up with other API's... let the programmers worry about that part. |
Danny, I think there is a misunderstanding here. We always only pass an opaque void* type to the renderer. ImGui has no knowledge or understanding of its content and we are not changing that.
The question here is whether we can change that opaque storage from sizeof(void*) to sizeof(u64) so we can accomodate for users wanting to a pass a 8 bytes value to their backend/renderer, and I am discussing the backward compatibility repercussion here.
|
If you are using modern api like Vulkan or DX12 in your engine it would also be useful to share how you are using ImTextureId with them (what are you storing in there?).
|
I understand, you still want to pass an opaque pointer and imGUI has no knowledge about it contents.. But what I was trying to say is that even tho ImGUI has no knowledge about it's contents.. it's Vulkan pointer is still there.. I think it's just wrong to mix raw Vulkan/dx12 (even opaque) pointers inside imGUI. In Vulkan you can use binding sets, so in my engine every Texture has it's own GUID. The graphics pipeline holds e.g 16 inputs with different set numbers (could start with dummy textures). In my case i'll probably never need more then 16 images at once in imGUI anyway. These dummy textures can be replaced by updating descriptors in your graphics pipeline with the ones you need. When you bake the command buffers you have VkBindDescriptorSets() .. link the correct set in engine (texture -> set ID) to set 0 in your GLSL shader which is normally the Font texture. When you need custom texture (which comes from imTexture) set different binding else just use your Font binding. This way there is no need to pass on Vulkan opaque pointers at all.. just a simple GUID. |
I vote for
In many cases I've seen the user uses shared_ptr or something similar which should be alive until the render func is called. We are currently doing stuff like wrapping the public imgui functions which take a texture to take a shared_ptr and then store the shared_ptr in some container which is cleared after the call to the draw func which is tedious. Having it a define would allow the user to make it whatever he wants and would eliminate the need to wrap. This illustrates my problem
|
This doesn't really fully solve the problem discussed here nor in #301 or #914, but as several people suggested it and it's harmless, I've added:
So it's possible to define ImTextureId in imconfig.h
This is a design choice and imgui or this thread isn't the place to decide that people shouldn't use pointers in their own engine. It's perfectly fine to do it one way or another. There are hundreds of ways to write engines and people can manage the lifetime of their pointers just fine.?
To me your example illustrate a non-problem since you just solved it? You can add a 2 lines functions to take care of handling the lifetime however you want, isn't that good enough? What's tedious about it? You could probably even wrap it at a lower-level:
So when doing imgui calls you can wrap your parameters directly without having to wrap anything else?
@thedmd I'd be curious if you could expand on the idea .
That's similar to the PR in #914. Everyone's throwing suggestions which I don't comprehend at the moment and I think I'll take a day and learn about Vulkan/DX12 before coming back to this topic. EDIT Found a new friend! https://vulkan-tutorial.com/ |
What I had in mind is simple texture API implementation, like one here: imgui_impl_dx11.cpp#L831 Basic version allows to create ARGB texture from user provided data: IMGUI_API ImTextureID ImGui_CreateTexture32(const void* data, int width, int height);
IMGUI_API void ImGui_DestroyTexture(ImTextureID texture);
IMGUI_API int ImGui_GetTextureWidth(ImTextureID texture);
IMGUI_API int ImGui_GetTextureHeight(ImTextureID texture); Different backends may provide additional versions of IMGUI_API ImTextureID ImGui_CreateTextureFromView(ID3D11ShaderResourceView* view); Beside abstracting away texture management this code serve as an example how one can deal with ImTextureID being a transparent pointer. Personally I used this to quickly feed test code with images, by mixing it with 'stb_image' to create: IMGUI_API ImTextureID ImGui_LoadTexture(const char* path); |
I've defined custom There's one problem, which can't be solved easily, though. Look at this. Here, textureId is being printed via "%p" which triggers warning when texture id = unsigned int. The only solution I see is to allow users to define custom format literal as it was done in inttypes.h for intptr_t, for example. So, we can have something like this: #ifndef ImTextureIDPrintLiteral
#define ImTextureIDPrintLiteral "%p"
#endif and in my case, I'll do: #define ImTextureIDPrintLiteral "%u" and in code it'll be something like: "Draw %4d %s vtx, tex " ImTextureIDPrintLiteral ", clip_rect (%4.0f,%4.0f)-(%4.0f,%4.0f)" Any thoughts on this? |
FYI i went forward and made ImTextureID default to ImU64 with 92b9498. I have tweaked the FAQ accordingly This was not a 100% necessary change but it's sane and it allows simplifying some build scripts/projects to avoid a define, which in turns makes things simpler to grasp for many. Note that you can always #define ImTextureID to a higher-level structure if you like, with custom constructors, holding more info such as sampling state etc. |
Both DirectX12 (#301) and Vulkan (#914) uses descriptors that are 64-bits.
This is a problem for our demos because ImTextureID is defined as pointer-size (
typedef void* ImTextureID
) and therefore on 32-bit target ImTextureID is 32-bits. This makes it difficult to use it as a convenient way to pass a texture descriptor. (in the case that the user wants to use a native description inside ImTextureID)I believe it is not a suuuper critical problem because:
MyTexture*
orMyMaterial*
into ImTextureID which wouldn't be a problem any more. Though some engine may use 64-bit integer identifiers in all build mode?(Let me know if you think those assumptions are wrong!)
So I think this mostly affect: (apart from our demo code)
One solution is to make ImTextureID always 64-bit. However C++ doesn't have a "pointer type that's always 64-bit" so it would have to be an integer and that mess up with some existing code.
(A) If I try that:
typedef ImU64 ImTextureID;
The kind of errors I get:
From imgui examples: (~7 errors)
Another codebase: (~40 errors)
(B) We can minimize the errors by using a higher-level primitive: (not really in the spirit of imgui's codebase, but let's try)
Then we get from imgui codebase (down from ~7 to 1 error)
Other codebase: (down from ~20 to 5 errors)
None of those solution are perfect and we'd creating a new problem for people updating to a new version. They are fixable issues, so not end of the world, but a little annoying considering it'd for a change that would benefit few people.
We'd also have to cast
MyTexture*
intoImTextureID
explicitly now, whereas previously it worke implicitly because it would cast to void*.(C)
Add an optional imconfig.h directive to make ImTextureID unsigned long long, add static_assert in the dx12/vulkan back-ends in 32-bit mode to refer to this option.
(D)
Specifically for Vulkan or DirectX12 decide that the binding can use another type, e.g. pass
VkDescriptorSet*
instead ofVkDescriptorSet
andD3D12_GPU_DESCRIPTOR_HANDLE*
instead ofD3D12_GPU_DESCRIPTOR_HANDLE
. I personally don't know either API at that point and don't know what would be best. Heck, I don't even understand what those things are, what their lifetime is in a typical dx12/vulkan codebase (there is an open question in #914 to decide how ImTextureID should be interpreted in the default Vulkan binding)(E)
Hide the issue under the carpet by removing 32-bit build for the Vulkan and DirectX12 demo from our example projects and/or upcoming project generator, leave the static asserts on.
(F)
Other suggestions?
The text was updated successfully, but these errors were encountered: