Skip to content
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

Exposing some hidden CFunctionReflection members to the front end (ID3D12LibraryReflection/ID3D12FunctionReflection) #135

Open
wants to merge 29 commits into
base: main
Choose a base branch
from

Conversation

Nielsbishere
Copy link

@Nielsbishere Nielsbishere commented Jul 25, 2024

The DXC backend has the following parameter:
const DxilFunctionProps *m_pProps
Which has a large number of members that are not accessible by ID3D12FunctionReflection and D3D12_FUNCTION_DESC.
The most interesting ones being raytracing and workgraph related.
Exposing these saves me a lot of headache, as I won't need to parse these manually, especially since they're already available in DXC.

The following parameters still need to be exposed (and I'll take a look later to add them):

  • Serialized root signature could have a getter which returns a DxcBuffer.
  • InputNodes / OutputNodes could have a getter which returns a CNodeIOProperties* and length. Where each CNodeIOProperties has const char* for name as well as UINT id.
  • NodeID NodeShaderId, NodeShaderSharedInput could use UINT ids in the D3D12_NODE_SHADER_DESC struct, while leaving the strings as getters.
  • D3D12_SHADER_DESC might need an update to provide similar functionality.

The following parameters I won't add, since I have no idea how to properly expose them and if they're even useful to the front end:

  • Vertex reflection: clipPlanes[6] returns an llvm::Constant
  • Hull reflection: returns patchConstFunction as llvm::Function, which won't be accessible.

Since these are llvm internal values, I'm guessing that they might not be very useful for the front end.

This relates to a DXCompiler issue & PR.

@jenatali
Copy link
Member

jenatali commented Jul 25, 2024

This relates to a DXCompiler issue & PR.

Please link the actual PR?

Copy link
Member

@jenatali jenatali left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I don't think we can merge a change to the D3D12_SHADER_DESC structure, since that structure is returned from both DXC and FXC, and FXC has versioning constraints that can't break. I think we can add new structures to this header, assuming the DXC changes are reviewed and accepted.

@Nielsbishere
Copy link
Author

Okay sounds good, I'll add an extended version that can only be queried for DXC. The DXC version is on my fork, but I gotta rebase it first, since it's built on other changes.

@Nielsbishere
Copy link
Author

Implemented by microsoft/DirectXShaderCompiler#6827 on the DXC side. Not sure if there's anything that has to be done for FXC? Maybe it shouldn't be declared "PURE" and return failed instead?

@jenatali
Copy link
Member

Implemented by microsoft/DirectXShaderCompiler#6827 on the DXC side. Not sure if there's anything that has to be done for FXC? Maybe it shouldn't be declared "PURE" and return failed instead?

The way to do this without breaking FXC is to put it on an ID3D12FunctionReflection1 with a new IID, and then you can implement QueryInterface for it.

Specifically, the layout of the virtual function table cannot change. Inserting a function in the middle of the table means that when you call that function, or later, on a concrete implementation coming from FXC or old DXC, you're calling a different function than you think you are. You could add it to the end, but then if you try to call it on an old concrete implementation, you'll just crash because there's nothing there in the function table. Even if you have a default implementation instead of PURE, you'd need to recompile the caller, for it to have a function at that spot in the vtable.

The only way to have a clean runtime failure is to add a separate interface and query.

@Nielsbishere
Copy link
Author

Nielsbishere commented Jul 26, 2024

It seems like I have to expose this with ID3D12LibraryReflection1::GetFunctionByIndex1, as ID3D12FunctionReflection doesn't inherrit IUnknown, which I don't think I can safely change. I can make the new ID3D12FunctionReflection1 inherit IUnknown for the future though... But not ideal any way.

Copy link
Member

@damyanp damyanp left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

There's quite a bit of API surface here that needs more careful review. So far, I've only looked at things that could be done to make the review process easier.

include/directx/d3d12shader.h Outdated Show resolved Hide resolved
include/directx/d3d12shader.h Outdated Show resolved Hide resolved
include/directx/d3d12shader.h Outdated Show resolved Hide resolved
…nputNode & GetOutputNode to ID3D12FunctionReflection1 to be able to be queried for Workgraphs. Added input/output node count and shaderId/shaderSharedInput (name & id) to D3D12_NODE_DESC
@Nielsbishere
Copy link
Author

Implemented PR changes. Also exposed all remaining properties such as node info and serialized root signature.
Only thing left to do could be implementing this in D3D12_SHADER_DESC1 as well. But that could be a PR for later.

Copy link
Member

@jenatali jenatali left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I'll let the DXC folks drive whether the corresponding implementation PR in the compiler repo is appropriate, but looking solely at the API surface exposed here, this seems acceptable to me.

@damyanp
Copy link
Member

damyanp commented Jul 31, 2024

See microsoft/DirectXShaderCompiler#6827 (comment)

We need to make sure we've completed a thorough review of all of this new API surface that'll need to be supported for a long time before accepting the change to the headers.

Thanks!

@jenatali
Copy link
Member

Yes, agreed, I have no intentions of actually merging this until the corresponding DXC change is reviewed and completed. Assuming that the implementation is accepted, we can take this into the headers and the Windows SDK afterwards.

…3D12ShaderReflection. Added GetWaveSize to query wave size.
@@ -416,6 +608,10 @@ DECLARE_INTERFACE_(ID3D12ShaderReflection, IUnknown)
_Out_opt_ UINT* pSizeZ) PURE;

STDMETHOD_(UINT64, GetRequiresFlags)(THIS) PURE;

STDMETHOD_(BOOL, GetWaveSize)
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This kind of breaking change made sense with the FXC model, where you linked against and shipped a redist D3DCompiler_xx.dll, and also makes sense with a strictly-DXC model. But in a world where D3DCompiler_47.dll is inbox in Windows and implements the old interface, where DXC can implement a new one, I think I'd prefer to rev this in a non-breaking way.

We can do that by just adding ID3D12ShaderReflection1, which inherits from ID3D12ShaderReflection and adds the new method. The VTable of ID3D12ShaderReflection1 is then identical to this new definition, except that apps can still query for the older one to avoid it being a source-breaking change for FXC consumers.

Copy link
Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Okay sounds good, I was planning on doing it that way, but assumed the comment was the way to do it. Guess I'lll implement it the right way.

…Size is missing as there's still an issue pending for that.
…C uses the assumption that bool* == LPBOOL, which isn't possible if BOOL is uint32_t, this needs some further attention to see if this is OK or has side effects, but DXC and DirectX12-Headers definitely don't agree on the size of a bool... Checking the define for LLVM_SUPPORT_WIN_ADAPTER_H is used to avoid duplicate defines and structs/typedefs, this is required on Linux since basestd.h from d3dcommon.h has its own definitions that are definitely different than from DXC. Made InlineIsEqualGUID equal to the implementation in DXC, since that one doesn't spam warnings.
@Nielsbishere
Copy link
Author

Nielsbishere commented Aug 7, 2024

Since last time, added numthreads to mesh/amplification shaders and implemented ID3D12ShaderReflection1.
There were some... issues with upgrading DXC to the latest DirectX-Headers, since it was behind by 2 years.

Important notes: DXC uses the assumption that bool* == LPBOOL, which isn't possible if BOOL is uint32_t, this needs some further attention to see if this is OK or has side effects, but DXC and DirectX12-Headers definitely don't agree on the size of a bool... Checking the define for LLVM_SUPPORT_WIN_ADAPTER_H is used to avoid duplicate defines and structs/typedefs, this is required on Linux since basestd.h from d3dcommon.h has its own definitions that are definitely different than from DXC (dxc/WinAdapter.h).

EDIT: Going to see if I can find a way to keep BOOL as uint32_t, since it might already be serialized like that?
EDIT2: I don't think so....

… 2 years behind. Removed IN/OUT and interface since they interfere with lots of files. Made REFGUID/REFCLSID/REFIID typedefs, since these defines interfere with everything too. Turned IsEqualGUID into dxc's version, to avoid lots of warnings.
@Nielsbishere
Copy link
Author

This branch is based of #137 to allow DXC to update to latest DirectX-Headers. DirectX-Headers itself should be fine now, just need to add support for the reflection data to dxcdisassembler and update unit tests to verify dxa output.

@Nielsbishere
Copy link
Author

Added an important function to ID3D12FunctionReflection1; which is GetRequiresFlags such as in ID3D12ShaderReflection. Without this, it is impossible to find out what features are being used by a lib file (I thought the desc Flags was used for this, but nope).

This is especially important now that some GPUs don't always support the base featureset, so I use the following to check feature support of the underlying GPUs:

		D3D_SHADER_REQUIRES_RAYTRACING_TIER_1_1,
		D3D_SHADER_REQUIRES_NATIVE_16BIT_OPS,
		D3D_SHADER_REQUIRES_INT64_OPS,
		D3D_SHADER_REQUIRES_VIEW_ID,
		D3D_SHADER_REQUIRES_DOUBLES,
		D3D_SHADER_REQUIRES_11_1_DOUBLE_EXTENSIONS,
		D3D_SHADER_REQUIRES_ATOMIC_INT64_ON_TYPED_RESOURCE,
		D3D_SHADER_REQUIRES_ATOMIC_INT64_ON_GROUP_SHARED,
		D3D_SHADER_REQUIRES_DERIVATIVES_IN_MESH_AND_AMPLIFICATION_SHADERS,
		D3D_SHADER_REQUIRES_WRITEABLE_MSAA_TEXTURES,
		D3D_SHADER_REQUIRES_WAVE_OPS

I use this reflection to know whether a shader is really using a feature. It's possible someone enables I64 to include something that requires it but never actually uses I64 in the final shader. This is then used to retroactively disable that feature to ensure we don't go checking features that aren't really relevant.

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

Successfully merging this pull request may close these issues.

4 participants