summaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorTim Foley <tfoleyNV@users.noreply.github.com>2017-11-01 08:30:45 -0700
committerGitHub <noreply@github.com>2017-11-01 08:30:45 -0700
commitec41631032b65973e8f92348e0a86bb9924ef981 (patch)
treee047fec206bb9fc3ed6303c6963b9e7207097d3a
parente2b473060133ae25a0be92fcfe80c5c7e51fa10a (diff)
Allow use of dxc compiler for DXIL generation (#241)
- Add shader model 6.0, 6.1, and 6.2 targets - Add DXIL and DXIL assembly as output formats - Add header for DXC API to `external/` - Add `dxc-support.cpp` that wraps usage of the API - Add `-pass-through dxc` option, equivalent to what we have for `fxc` Notes: * This does *not* include any logic to add `dxcompiler.dll` to our build process; that is way out of scope for the build complexity I'm ready to deal with * For right now, the use of `dxcompiler.dll` is hard-coded, and it must be discoverable in the current executable's search path; options to customize can come later * The `-pass-through` option is kind of silly because the code doesn't actually pay attention to the value (just whether it is set). If you set it to `fxc` but ask for DXIL, we pass through `dxc` anyway.
-rw-r--r--external/dxc/dxcapi.h368
-rw-r--r--slang.h3
-rw-r--r--source/slang/compiler.cpp83
-rw-r--r--source/slang/compiler.h7
-rw-r--r--source/slang/dxc-support.cpp280
-rw-r--r--source/slang/options.cpp3
-rw-r--r--source/slang/profile-defs.h29
-rw-r--r--source/slang/slang.vcxproj1
-rw-r--r--source/slang/slang.vcxproj.filters1
-rw-r--r--source/slang/type-layout.cpp2
10 files changed, 773 insertions, 4 deletions
diff --git a/external/dxc/dxcapi.h b/external/dxc/dxcapi.h
new file mode 100644
index 000000000..70f93e834
--- /dev/null
+++ b/external/dxc/dxcapi.h
@@ -0,0 +1,368 @@
+
+///////////////////////////////////////////////////////////////////////////////
+// //
+// dxcapi.h //
+// Copyright (C) Microsoft Corporation. All rights reserved. //
+// This file is distributed under the University of Illinois Open Source //
+// License. See LICENSE.TXT for details. //
+// //
+// Provides declarations for the DirectX Compiler API entry point. //
+// //
+///////////////////////////////////////////////////////////////////////////////
+
+#ifndef __DXC_API__
+#define __DXC_API__
+
+#ifndef DXC_API_IMPORT
+#define DXC_API_IMPORT __declspec(dllimport)
+#endif
+
+struct IMalloc;
+struct IDxcIncludeHandler;
+
+/// <summary>
+/// Creates a single uninitialized object of the class associated with a specified CLSID.
+/// </summary>
+/// <param name="rclsid">
+/// The CLSID associated with the data and code that will be used to create the object.
+/// </param>
+/// <param name="riid">
+/// A reference to the identifier of the interface to be used to communicate
+/// with the object.
+/// </param>
+/// <param name="ppv">
+/// Address of pointer variable that receives the interface pointer requested
+/// in riid. Upon successful return, *ppv contains the requested interface
+/// pointer. Upon failure, *ppv contains NULL.</param>
+/// <remarks>
+/// While this function is similar to CoCreateInstance, there is no COM involvement.
+/// </remarks>
+typedef HRESULT (__stdcall *DxcCreateInstanceProc)(
+ _In_ REFCLSID rclsid,
+ _In_ REFIID riid,
+ _Out_ LPVOID* ppv
+);
+
+typedef HRESULT(__stdcall *DxcCreateInstance2Proc)(
+ _In_ IMalloc *pMalloc,
+ _In_ REFCLSID rclsid,
+ _In_ REFIID riid,
+ _Out_ LPVOID* ppv
+ );
+
+/// <summary>
+/// Creates a single uninitialized object of the class associated with a specified CLSID.
+/// </summary>
+/// <param name="rclsid">
+/// The CLSID associated with the data and code that will be used to create the object.
+/// </param>
+/// <param name="riid">
+/// A reference to the identifier of the interface to be used to communicate
+/// with the object.
+/// </param>
+/// <param name="ppv">
+/// Address of pointer variable that receives the interface pointer requested
+/// in riid. Upon successful return, *ppv contains the requested interface
+/// pointer. Upon failure, *ppv contains NULL.</param>
+/// <remarks>
+/// While this function is similar to CoCreateInstance, there is no COM involvement.
+/// </remarks>
+DXC_API_IMPORT HRESULT __stdcall DxcCreateInstance(
+ _In_ REFCLSID rclsid,
+ _In_ REFIID riid,
+ _Out_ LPVOID* ppv
+ );
+
+DXC_API_IMPORT HRESULT __stdcall DxcCreateInstance2(
+ _In_ IMalloc *pMalloc,
+ _In_ REFCLSID rclsid,
+ _In_ REFIID riid,
+ _Out_ LPVOID* ppv
+);
+
+
+// IDxcBlob is an alias of ID3D10Blob and ID3DBlob
+struct __declspec(uuid("8BA5FB08-5195-40e2-AC58-0D989C3A0102"))
+IDxcBlob : public IUnknown {
+public:
+ virtual LPVOID STDMETHODCALLTYPE GetBufferPointer(void) = 0;
+ virtual SIZE_T STDMETHODCALLTYPE GetBufferSize(void) = 0;
+};
+
+struct __declspec(uuid("7241d424-2646-4191-97c0-98e96e42fc68"))
+IDxcBlobEncoding : public IDxcBlob {
+public:
+ virtual HRESULT STDMETHODCALLTYPE GetEncoding(_Out_ BOOL *pKnown,
+ _Out_ UINT32 *pCodePage) = 0;
+};
+
+struct __declspec(uuid("e5204dc7-d18c-4c3c-bdfb-851673980fe7"))
+IDxcLibrary : public IUnknown {
+ virtual HRESULT STDMETHODCALLTYPE SetMalloc(_In_opt_ IMalloc *pMalloc) = 0;
+ virtual HRESULT STDMETHODCALLTYPE CreateBlobFromBlob(
+ _In_ IDxcBlob *pBlob, UINT32 offset, UINT32 length, _COM_Outptr_ IDxcBlob **ppResult) = 0;
+ virtual HRESULT STDMETHODCALLTYPE CreateBlobFromFile(
+ LPCWSTR pFileName, _In_opt_ UINT32* codePage,
+ _COM_Outptr_ IDxcBlobEncoding **pBlobEncoding) = 0;
+ virtual HRESULT STDMETHODCALLTYPE CreateBlobWithEncodingFromPinned(
+ LPBYTE pText, UINT32 size, UINT32 codePage,
+ _COM_Outptr_ IDxcBlobEncoding **pBlobEncoding) = 0;
+ virtual HRESULT STDMETHODCALLTYPE CreateBlobWithEncodingOnHeapCopy(
+ _In_bytecount_(size) LPCVOID pText, UINT32 size, UINT32 codePage,
+ _COM_Outptr_ IDxcBlobEncoding **pBlobEncoding) = 0;
+ virtual HRESULT STDMETHODCALLTYPE CreateBlobWithEncodingOnMalloc(
+ _In_bytecount_(size) LPCVOID pText, IMalloc *pIMalloc, UINT32 size, UINT32 codePage,
+ _COM_Outptr_ IDxcBlobEncoding **pBlobEncoding) = 0;
+ virtual HRESULT STDMETHODCALLTYPE CreateIncludeHandler(
+ _COM_Outptr_ IDxcIncludeHandler **ppResult) = 0;
+ virtual HRESULT STDMETHODCALLTYPE CreateStreamFromBlobReadOnly(
+ _In_ IDxcBlob *pBlob, _COM_Outptr_ IStream **ppStream) = 0;
+ virtual HRESULT STDMETHODCALLTYPE GetBlobAsUtf8(
+ _In_ IDxcBlob *pBlob, _COM_Outptr_ IDxcBlobEncoding **pBlobEncoding) = 0;
+ virtual HRESULT STDMETHODCALLTYPE GetBlobAsUtf16(
+ _In_ IDxcBlob *pBlob, _COM_Outptr_ IDxcBlobEncoding **pBlobEncoding) = 0;
+};
+
+struct __declspec(uuid("CEDB484A-D4E9-445A-B991-CA21CA157DC2"))
+IDxcOperationResult : public IUnknown {
+ virtual HRESULT STDMETHODCALLTYPE GetStatus(_Out_ HRESULT *pStatus) = 0;
+ virtual HRESULT STDMETHODCALLTYPE GetResult(_COM_Outptr_result_maybenull_ IDxcBlob **pResult) = 0;
+ virtual HRESULT STDMETHODCALLTYPE GetErrorBuffer(_COM_Outptr_result_maybenull_ IDxcBlobEncoding **pErrors) = 0;
+};
+
+struct __declspec(uuid("7f61fc7d-950d-467f-b3e3-3c02fb49187c"))
+IDxcIncludeHandler : public IUnknown {
+ virtual HRESULT STDMETHODCALLTYPE LoadSource(
+ _In_ LPCWSTR pFilename, // Candidate filename.
+ _COM_Outptr_result_maybenull_ IDxcBlob **ppIncludeSource // Resultant source object for included file, nullptr if not found.
+ ) = 0;
+};
+
+struct DxcDefine {
+ LPCWSTR Name;
+ _Maybenull_ LPCWSTR Value;
+};
+
+struct __declspec(uuid("8c210bf3-011f-4422-8d70-6f9acb8db617"))
+IDxcCompiler : public IUnknown {
+ // Compile a single entry point to the target shader model
+ virtual HRESULT STDMETHODCALLTYPE Compile(
+ _In_ IDxcBlob *pSource, // Source text to compile
+ _In_opt_ LPCWSTR pSourceName, // Optional file name for pSource. Used in errors and include handlers.
+ _In_ LPCWSTR pEntryPoint, // entry point name
+ _In_ LPCWSTR pTargetProfile, // shader profile to compile
+ _In_count_(argCount) LPCWSTR *pArguments, // Array of pointers to arguments
+ _In_ UINT32 argCount, // Number of arguments
+ _In_count_(defineCount) const DxcDefine *pDefines, // Array of defines
+ _In_ UINT32 defineCount, // Number of defines
+ _In_opt_ IDxcIncludeHandler *pIncludeHandler, // user-provided interface to handle #include directives (optional)
+ _COM_Outptr_ IDxcOperationResult **ppResult // Compiler output status, buffer, and errors
+ ) = 0;
+
+ // Preprocess source text
+ virtual HRESULT STDMETHODCALLTYPE Preprocess(
+ _In_ IDxcBlob *pSource, // Source text to preprocess
+ _In_opt_ LPCWSTR pSourceName, // Optional file name for pSource. Used in errors and include handlers.
+ _In_count_(argCount) LPCWSTR *pArguments, // Array of pointers to arguments
+ _In_ UINT32 argCount, // Number of arguments
+ _In_count_(defineCount) const DxcDefine *pDefines, // Array of defines
+ _In_ UINT32 defineCount, // Number of defines
+ _In_opt_ IDxcIncludeHandler *pIncludeHandler, // user-provided interface to handle #include directives (optional)
+ _COM_Outptr_ IDxcOperationResult **ppResult // Preprocessor output status, buffer, and errors
+ ) = 0;
+
+ // Disassemble a program.
+ virtual HRESULT STDMETHODCALLTYPE Disassemble(
+ _In_ IDxcBlob *pSource, // Program to disassemble.
+ _COM_Outptr_ IDxcBlobEncoding **ppDisassembly // Disassembly text.
+ ) = 0;
+};
+
+struct __declspec(uuid("A005A9D9-B8BB-4594-B5C9-0E633BEC4D37"))
+IDxcCompiler2 : public IDxcCompiler {
+ // Compile a single entry point to the target shader model with debug information.
+ virtual HRESULT STDMETHODCALLTYPE CompileWithDebug(
+ _In_ IDxcBlob *pSource, // Source text to compile
+ _In_opt_ LPCWSTR pSourceName, // Optional file name for pSource. Used in errors and include handlers.
+ _In_ LPCWSTR pEntryPoint, // Entry point name
+ _In_ LPCWSTR pTargetProfile, // Shader profile to compile
+ _In_count_(argCount) LPCWSTR *pArguments, // Array of pointers to arguments
+ _In_ UINT32 argCount, // Number of arguments
+ _In_count_(defineCount) const DxcDefine *pDefines, // Array of defines
+ _In_ UINT32 defineCount, // Number of defines
+ _In_opt_ IDxcIncludeHandler *pIncludeHandler, // user-provided interface to handle #include directives (optional)
+ _COM_Outptr_ IDxcOperationResult **ppResult, // Compiler output status, buffer, and errors
+ _Outptr_opt_result_z_ LPWSTR *ppDebugBlobName,// Suggested file name for debug blob.
+ _COM_Outptr_opt_ IDxcBlob **ppDebugBlob // Debug blob
+ ) = 0;
+};
+
+struct __declspec(uuid("F1B5BE2A-62DD-4327-A1C2-42AC1E1E78E6"))
+IDxcLinker : public IUnknown {
+public:
+ // Register a library with name to ref it later.
+ virtual HRESULT RegisterLibrary(
+ _In_opt_ LPCWSTR pLibName, // Name of the library.
+ _In_ IDxcBlob *pLib // Library blob.
+ ) = 0;
+
+ // Links the shader and produces a shader blob that the Direct3D runtime can
+ // use.
+ virtual HRESULT STDMETHODCALLTYPE Link(
+ _In_opt_ LPCWSTR pEntryName, // Entry point name
+ _In_ LPCWSTR pTargetProfile, // shader profile to link
+ _In_count_(libCount)
+ const LPCWSTR *pLibNames, // Array of library names to link
+ UINT32 libCount, // Number of libraries to link
+ _In_count_(argCount)
+ const LPCWSTR *pArguments, // Array of pointers to arguments
+ _In_ UINT32 argCount, // Number of arguments
+ _COM_Outptr_ IDxcOperationResult *
+ *ppResult // Linker output status, buffer, and errors
+ ) = 0;
+};
+
+static const UINT32 DxcValidatorFlags_Default = 0;
+static const UINT32 DxcValidatorFlags_InPlaceEdit = 1; // Validator is allowed to update shader blob in-place.
+static const UINT32 DxcValidatorFlags_RootSignatureOnly = 2;
+static const UINT32 DxcValidatorFlags_ModuleOnly = 4;
+static const UINT32 DxcValidatorFlags_ValidMask = 0x7;
+
+struct __declspec(uuid("A6E82BD2-1FD7-4826-9811-2857E797F49A"))
+IDxcValidator : public IUnknown {
+ // Validate a shader.
+ virtual HRESULT STDMETHODCALLTYPE Validate(
+ _In_ IDxcBlob *pShader, // Shader to validate.
+ _In_ UINT32 Flags, // Validation flags.
+ _COM_Outptr_ IDxcOperationResult **ppResult // Validation output status, buffer, and errors
+ ) = 0;
+};
+
+struct __declspec(uuid("334b1f50-2292-4b35-99a1-25588d8c17fe"))
+IDxcContainerBuilder : public IUnknown {
+ virtual HRESULT STDMETHODCALLTYPE Load(_In_ IDxcBlob *pDxilContainerHeader) = 0; // Loads DxilContainer to the builder
+ virtual HRESULT STDMETHODCALLTYPE AddPart(_In_ UINT32 fourCC, _In_ IDxcBlob *pSource) = 0; // Part to add to the container
+ virtual HRESULT STDMETHODCALLTYPE RemovePart(_In_ UINT32 fourCC) = 0; // Remove the part with fourCC
+ virtual HRESULT STDMETHODCALLTYPE SerializeContainer(_Out_ IDxcOperationResult **ppResult) = 0; // Builds a container of the given container builder state
+};
+
+struct __declspec(uuid("091f7a26-1c1f-4948-904b-e6e3a8a771d5"))
+IDxcAssembler : public IUnknown {
+ // Assemble dxil in ll or llvm bitcode to DXIL container.
+ virtual HRESULT STDMETHODCALLTYPE AssembleToContainer(
+ _In_ IDxcBlob *pShader, // Shader to assemble.
+ _COM_Outptr_ IDxcOperationResult **ppResult // Assembly output status, buffer, and errors
+ ) = 0;
+};
+
+struct __declspec(uuid("d2c21b26-8350-4bdc-976a-331ce6f4c54c"))
+IDxcContainerReflection : public IUnknown {
+ virtual HRESULT STDMETHODCALLTYPE Load(_In_ IDxcBlob *pContainer) = 0; // Container to load.
+ virtual HRESULT STDMETHODCALLTYPE GetPartCount(_Out_ UINT32 *pResult) = 0;
+ virtual HRESULT STDMETHODCALLTYPE GetPartKind(UINT32 idx, _Out_ UINT32 *pResult) = 0;
+ virtual HRESULT STDMETHODCALLTYPE GetPartContent(UINT32 idx, _COM_Outptr_ IDxcBlob **ppResult) = 0;
+ virtual HRESULT STDMETHODCALLTYPE FindFirstPartKind(UINT32 kind, _Out_ UINT32 *pResult) = 0;
+ virtual HRESULT STDMETHODCALLTYPE GetPartReflection(UINT32 idx, REFIID iid, void **ppvObject) = 0;
+};
+
+struct __declspec(uuid("AE2CD79F-CC22-453F-9B6B-B124E7A5204C"))
+IDxcOptimizerPass : public IUnknown {
+ virtual HRESULT STDMETHODCALLTYPE GetOptionName(_COM_Outptr_ LPWSTR *ppResult) = 0;
+ virtual HRESULT STDMETHODCALLTYPE GetDescription(_COM_Outptr_ LPWSTR *ppResult) = 0;
+ virtual HRESULT STDMETHODCALLTYPE GetOptionArgCount(_Out_ UINT32 *pCount) = 0;
+ virtual HRESULT STDMETHODCALLTYPE GetOptionArgName(UINT32 argIndex, _COM_Outptr_ LPWSTR *ppResult) = 0;
+ virtual HRESULT STDMETHODCALLTYPE GetOptionArgDescription(UINT32 argIndex, _COM_Outptr_ LPWSTR *ppResult) = 0;
+};
+
+struct __declspec(uuid("25740E2E-9CBA-401B-9119-4FB42F39F270"))
+IDxcOptimizer : public IUnknown {
+ virtual HRESULT STDMETHODCALLTYPE GetAvailablePassCount(_Out_ UINT32 *pCount) = 0;
+ virtual HRESULT STDMETHODCALLTYPE GetAvailablePass(UINT32 index, _COM_Outptr_ IDxcOptimizerPass** ppResult) = 0;
+ virtual HRESULT STDMETHODCALLTYPE RunOptimizer(IDxcBlob *pBlob,
+ _In_count_(optionCount) LPCWSTR *ppOptions, UINT32 optionCount,
+ _COM_Outptr_ IDxcBlob **pOutputModule,
+ _COM_Outptr_opt_ IDxcBlobEncoding **ppOutputText) = 0;
+};
+
+static const UINT32 DxcVersionInfoFlags_None = 0;
+static const UINT32 DxcVersionInfoFlags_Debug = 1; // Matches VS_FF_DEBUG
+static const UINT32 DxcVersionInfoFlags_Internal = 2; // Internal Validator (non-signing)
+
+struct __declspec(uuid("b04f5b50-2059-4f12-a8ff-a1e0cde1cc7e"))
+IDxcVersionInfo : public IUnknown {
+ virtual HRESULT STDMETHODCALLTYPE GetVersion(_Out_ UINT32 *pMajor, _Out_ UINT32 *pMinor) = 0;
+ virtual HRESULT STDMETHODCALLTYPE GetFlags(_Out_ UINT32 *pFlags) = 0;
+};
+
+// {73e22d93-e6ce-47f3-b5bf-f0664f39c1b0}
+__declspec(selectany) extern const CLSID CLSID_DxcCompiler = {
+ 0x73e22d93,
+ 0xe6ce,
+ 0x47f3,
+ { 0xb5, 0xbf, 0xf0, 0x66, 0x4f, 0x39, 0xc1, 0xb0 }
+};
+
+// {EF6A8087-B0EA-4D56-9E45-D07E1A8B7806}
+__declspec(selectany) extern const GUID CLSID_DxcLinker = {
+ 0xef6a8087,
+ 0xb0ea,
+ 0x4d56,
+ {0x9e, 0x45, 0xd0, 0x7e, 0x1a, 0x8b, 0x78, 0x6}
+};
+
+// {CD1F6B73-2AB0-484D-8EDC-EBE7A43CA09F}
+__declspec(selectany) extern const CLSID CLSID_DxcDiaDataSource = {
+ 0xcd1f6b73,
+ 0x2ab0,
+ 0x484d,
+ { 0x8e, 0xdc, 0xeb, 0xe7, 0xa4, 0x3c, 0xa0, 0x9f }
+};
+
+// {6245D6AF-66E0-48FD-80B4-4D271796748C}
+__declspec(selectany) extern const GUID CLSID_DxcLibrary = {
+ 0x6245d6af,
+ 0x66e0,
+ 0x48fd,
+ { 0x80, 0xb4, 0x4d, 0x27, 0x17, 0x96, 0x74, 0x8c }
+};
+
+// {8CA3E215-F728-4CF3-8CDD-88AF917587A1}
+__declspec(selectany) extern const GUID CLSID_DxcValidator = {
+ 0x8ca3e215,
+ 0xf728,
+ 0x4cf3,
+ { 0x8c, 0xdd, 0x88, 0xaf, 0x91, 0x75, 0x87, 0xa1 }
+};
+
+// {D728DB68-F903-4F80-94CD-DCCF76EC7151}
+__declspec(selectany) extern const GUID CLSID_DxcAssembler = {
+ 0xd728db68,
+ 0xf903,
+ 0x4f80,
+ { 0x94, 0xcd, 0xdc, 0xcf, 0x76, 0xec, 0x71, 0x51 }
+};
+
+// {b9f54489-55b8-400c-ba3a-1675e4728b91}
+__declspec(selectany) extern const GUID CLSID_DxcContainerReflection = {
+ 0xb9f54489,
+ 0x55b8,
+ 0x400c,
+ { 0xba, 0x3a, 0x16, 0x75, 0xe4, 0x72, 0x8b, 0x91 }
+};
+
+// {AE2CD79F-CC22-453F-9B6B-B124E7A5204C}
+__declspec(selectany) extern const GUID CLSID_DxcOptimizer = {
+ 0xae2cd79f,
+ 0xcc22,
+ 0x453f,
+ {0x9b, 0x6b, 0xb1, 0x24, 0xe7, 0xa5, 0x20, 0x4c}
+};
+
+// {94134294-411f-4574-b4d0-8741e25240d2}
+__declspec(selectany) extern const GUID CLSID_DxcContainerBuilder = {
+ 0x94134294,
+ 0x411f,
+ 0x4574,
+ { 0xb4, 0xd0, 0x87, 0x41, 0xe2, 0x52, 0x40, 0xd2 }
+};
+#endif
diff --git a/slang.h b/slang.h
index 11b34053c..5c84ce6f3 100644
--- a/slang.h
+++ b/slang.h
@@ -89,6 +89,8 @@ extern "C"
SLANG_SPIRV_ASM,
SLANG_DXBC,
SLANG_DXBC_ASM,
+ SLANG_DXIL,
+ SLANG_DXIL_ASM,
};
/* A "container format" describes the way that the outputs
@@ -1079,6 +1081,7 @@ namespace slang
#include "source/core/text-io.cpp"
#include "source/slang/bytecode.cpp"
#include "source/slang/diagnostics.cpp"
+#include "source/slang/dxc-support.cpp"
#include "source/slang/emit.cpp"
#include "source/slang/ir.cpp"
#include "source/slang/lexer.cpp"
diff --git a/source/slang/compiler.cpp b/source/slang/compiler.cpp
index 8cf801a79..f045253b0 100644
--- a/source/slang/compiler.cpp
+++ b/source/slang/compiler.cpp
@@ -15,7 +15,7 @@
#include "reflection.h"
#include "emit.h"
-// Enable calling through to `fxc` to
+// Enable calling through to `fxc` or `dxc` to
// generate code on Windows.
#ifdef _WIN32
#define WIN32_LEAN_AND_MEAN
@@ -27,12 +27,18 @@
#ifndef SLANG_ENABLE_DXBC_SUPPORT
#define SLANG_ENABLE_DXBC_SUPPORT 1
#endif
+ #ifndef SLANG_ENABLE_DXIL_SUPPORT
+ #define SLANG_ENABLE_DXIL_SUPPORT 1
+ #endif
#endif
//
-// Otherwise, don't enable DXBC by default:
+// Otherwise, don't enable DXBC/DXIL by default:
#ifndef SLANG_ENABLE_DXBC_SUPPORT
#define SLANG_ENABLE_DXBC_SUPPORT 0
#endif
+#ifndef SLANG_ENABLE_DXIL_SUPPORT
+ #define SLANG_ENABLE_DXIL_SUPPORT 0
+#endif
// Enable calling through to `glslang` on
// all platforms.
@@ -356,6 +362,22 @@ namespace Slang
}
#endif
+#if SLANG_ENABLE_DXIL_SUPPORT
+
+// Implementations in `dxc-support.cpp`
+
+int emitDXILForEntryPointUsingDXC(
+ EntryPointRequest* entryPoint,
+ TargetRequest* targetReq,
+ List<uint8_t>& outCode);
+
+String dissassembleDXILUsingDXC(
+ CompileRequest* compileRequest,
+ void const* data,
+ size_t size);
+
+#endif
+
#if SLANG_ENABLE_GLSLANG_SUPPORT
SharedLibrary loadGLSLCompilerDLL(CompileRequest* request)
@@ -540,6 +562,38 @@ namespace Slang
break;
#endif
+#if SLANG_ENABLE_DXIL_SUPPORT
+ case CodeGenTarget::DXIL:
+ {
+ List<uint8_t> code;
+ int err = emitDXILForEntryPointUsingDXC(entryPoint, targetReq, code);
+ if (!err)
+ {
+ maybeDumpIntermediate(compileRequest, code.Buffer(), code.Count(), target);
+ result = CompileResult(code);
+ }
+ }
+ break;
+
+ case CodeGenTarget::DXILAssembly:
+ {
+ List<uint8_t> code;
+ int err = emitDXILForEntryPointUsingDXC(entryPoint, targetReq, code);
+ if (!err)
+ {
+ String assembly = dissassembleDXILUsingDXC(
+ compileRequest,
+ code.Buffer(),
+ code.Count());
+
+ maybeDumpIntermediate(compileRequest, assembly.Buffer(), target);
+
+ result = CompileResult(assembly);
+ }
+ }
+ break;
+#endif
+
case CodeGenTarget::SPIRV:
{
List<uint8_t> code = emitSPIRVForEntryPoint(entryPoint, targetReq);
@@ -703,6 +757,17 @@ namespace Slang
break;
#endif
+ #if SLANG_ENABLE_DXIL_SUPPORT
+ case CodeGenTarget::DXIL:
+ {
+ String assembly = dissassembleDXILUsingDXC(compileRequest,
+ data.begin(),
+ data.end() - data.begin());
+ writeOutputToConsole(compileRequest, assembly);
+ }
+ break;
+ #endif
+
case CodeGenTarget::SPIRV:
{
String assembly = dissassembleSPIRV(compileRequest,
@@ -943,6 +1008,20 @@ namespace Slang
}
break;
#endif
+
+ #if SLANG_ENABLE_DXIL_SUPPORT
+ case CodeGenTarget::DXILAssembly:
+ dumpIntermediateText(compileRequest, data, size, ".dxil.asm");
+ break;
+
+ case CodeGenTarget::DXIL:
+ dumpIntermediateBinary(compileRequest, data, size, ".dxil");
+ {
+ String dxilAssembly = dissassembleDXILUsingDXC(compileRequest, data, size);
+ dumpIntermediateText(compileRequest, dxilAssembly.begin(), dxilAssembly.Length(), ".dxil.asm");
+ }
+ break;
+ #endif
}
}
diff --git a/source/slang/compiler.h b/source/slang/compiler.h
index b7ab980fc..7b539ce69 100644
--- a/source/slang/compiler.h
+++ b/source/slang/compiler.h
@@ -47,6 +47,8 @@ namespace Slang
SPIRVAssembly = SLANG_SPIRV_ASM,
DXBytecode = SLANG_DXBC,
DXBytecodeAssembly = SLANG_DXBC_ASM,
+ DXIL = SLANG_DXIL,
+ DXILAssembly = SLANG_DXIL_ASM,
};
enum class ContainerFormat
@@ -126,8 +128,9 @@ namespace Slang
enum class PassThroughMode : SlangPassThrough
{
None = SLANG_PASS_THROUGH_NONE, // don't pass through: use Slang compiler
- HLSL = SLANG_PASS_THROUGH_FXC, // pass through HLSL to `D3DCompile` API
-// GLSL, // pass through GLSL to `glslang` library
+ fxc = SLANG_PASS_THROUGH_FXC, // pass through HLSL to `D3DCompile` API
+ dxc = SLANG_PASS_THROUGH_DXC, // pass through HLSL to `IDxcCompiler` API
+ glslang = SLANG_PASS_THROUGH_GLSLANG, // pass through GLSL to `glslang` library
};
class SourceFile;
diff --git a/source/slang/dxc-support.cpp b/source/slang/dxc-support.cpp
new file mode 100644
index 000000000..8de63f494
--- /dev/null
+++ b/source/slang/dxc-support.cpp
@@ -0,0 +1,280 @@
+// dxc-support.cpp
+#include "compiler.h"
+
+// This file implements support for invoking the `dxcompiler`
+// library to translate HLSL to DXIL.
+
+#if defined(_WIN32)
+# if !defined(SLANG_ENABLE_DXIL_SUPPORT)
+# define SLANG_ENABLE_DXIL_SUPPORT 1
+# endif
+#endif
+
+#if !defined(SLANG_ENABLE_DXIL_SUPPORT)
+# define SLANG_ENABLE_DXIL_SUPPORT 0
+#endif
+
+#if SLANG_ENABLE_DXIL_SUPPORT
+
+#define WIN32_LEAN_AND_MEAN
+#define NOMINMAX
+#include <Windows.h>
+#include <Unknwn.h>
+#include "../../external/dxc/dxcapi.h"
+#undef WIN32_LEAN_AND_MEAN
+#undef NOMINMAX
+
+#include "../core/platform.h"
+
+namespace Slang
+{
+ char const* GetHLSLProfileName(Profile profile);
+ String emitHLSLForEntryPoint(
+ EntryPointRequest* entryPoint,
+ TargetRequest* targetReq);
+
+ SharedLibrary loadDXCSharedLibrary(CompileRequest* request)
+ {
+ // TODO(tfoley): Let user specify name/path of library to use.
+ char const* libraryName = "dxcompiler";
+
+ SharedLibrary library = SharedLibrary::load(libraryName);
+ if (!library)
+ {
+ request->mSink.diagnose(SourceLoc(), Diagnostics::failedToLoadDynamicLibrary, libraryName);
+ }
+ return library;
+ }
+
+ SharedLibrary getDXCSharedLibrary(CompileRequest* request)
+ {
+ static SharedLibrary library = loadDXCSharedLibrary(request);
+ return library;
+ }
+
+ int emitDXILForEntryPointUsingDXC(
+ EntryPointRequest* entryPoint,
+ TargetRequest* targetReq,
+ List<uint8_t>& outCode)
+ {
+ auto compileRequest = entryPoint->compileRequest;
+
+ // First deal with all the rigamarole of loading
+ // the `dxcompiler` library, and creating the
+ // top-level COM objects that will be used to
+ // compile things.
+
+ static DxcCreateInstanceProc dxcCreateInstance = nullptr;
+ if (!dxcCreateInstance)
+ {
+ auto dxcSharedLibrary = getDXCSharedLibrary(compileRequest);
+ if (!dxcSharedLibrary)
+ return 1;
+
+ dxcCreateInstance = (DxcCreateInstanceProc) dxcSharedLibrary.findFuncByName("DxcCreateInstance");
+ if (!dxcCreateInstance)
+ return 1;
+ }
+
+ IDxcCompiler* dxcCompiler = nullptr;
+ if (FAILED(dxcCreateInstance(
+ CLSID_DxcCompiler,
+ __uuidof(dxcCompiler),
+ (LPVOID*) &dxcCompiler)))
+ {
+ return 1;
+ }
+
+ IDxcLibrary* dxcLibrary = nullptr;
+ if (FAILED(dxcCreateInstance(
+ CLSID_DxcLibrary,
+ __uuidof(dxcLibrary),
+ (LPVOID*) &dxcLibrary)))
+ {
+ return 1;
+ }
+
+ // Now let's go ahead and generate HLSL for the entry
+ // point, since we'll need that to feed into dxc.
+ auto hlslCode = emitHLSLForEntryPoint(entryPoint, targetReq);
+ maybeDumpIntermediate(entryPoint->compileRequest, hlslCode.Buffer(), CodeGenTarget::HLSL);
+
+ // Wrap the
+
+ // Create blob from the string
+ IDxcBlobEncoding* dxcSourceBlob = nullptr;
+ if (FAILED(dxcLibrary->CreateBlobWithEncodingFromPinned(
+ (LPBYTE)hlslCode.Buffer(),
+ (UINT32)hlslCode.Length(),
+ 0,
+ &dxcSourceBlob)))
+ {
+ return 1;
+ }
+
+ WCHAR const* args[16];
+ UINT32 argCount = 0;
+
+ // TODO: deal with
+ bool treatWarningsAsErrors = false;
+ if (treatWarningsAsErrors)
+ {
+ args[argCount++] = L"-WX";
+ }
+
+ String entryPointName = getText(entryPoint->name);
+ OSString wideEntryPointName = entryPointName.ToWString();
+
+ String profileName = GetHLSLProfileName(entryPoint->profile);
+ OSString wideProfileName = profileName.ToWString();
+
+
+ IDxcOperationResult* dxcResult = nullptr;
+ if (FAILED(dxcCompiler->Compile(dxcSourceBlob,
+ L"slang",
+ wideEntryPointName.begin(),
+ wideProfileName.begin(),
+ args,
+ argCount,
+ nullptr, // `#define`s
+ 0, // `#define` count
+ nullptr, // `#include` handler
+ &dxcResult)))
+ {
+ return 1;
+ }
+
+ // Retrieve result.
+ HRESULT resultCode = S_OK;
+ if (FAILED(dxcResult->GetStatus(&resultCode)))
+ {
+ // This indicates that we failed to retrieve the reuslt...
+ return 1;
+ }
+
+ // Note: it seems like the dxcompiler interface
+ // doesn't support querying diagnostic output
+ // *unless* the compile failed (no way to get
+ // warnings out!?).
+
+ // Verify compile result
+ if (FAILED(resultCode))
+ {
+ // Compilation failed.
+
+
+ // Try to read any diagnostic output.
+ IDxcBlobEncoding* dxcErrorBlob = nullptr;
+ if (!FAILED(dxcResult->GetErrorBuffer(&dxcErrorBlob)))
+ {
+ void* data = dxcErrorBlob->GetBufferPointer();
+ compileRequest->mSink.diagnoseRaw(
+ FAILED(resultCode) ? Severity::Error : Severity::Warning,
+ (char const*)dxcErrorBlob->GetBufferPointer());
+ dxcErrorBlob->Release();
+ }
+
+
+ return 1;
+ }
+
+ // Okay, the compile supposedly succeeded, so we
+ // just need to grab the buffer with the output DXIL.
+ IDxcBlob* dxcResultBlob = nullptr;
+ if (FAILED(dxcResult->GetResult(&dxcResultBlob)))
+ {
+ return 1;
+ }
+
+ outCode.AddRange(
+ (uint8_t const*)dxcResultBlob->GetBufferPointer(),
+ (int) dxcResultBlob->GetBufferSize());
+
+ // Clean up after ourselves.
+
+ if(dxcResultBlob) dxcResultBlob ->Release();
+ if(dxcResult) dxcResult ->Release();
+ if(dxcLibrary) dxcLibrary ->Release();
+ if(dxcCompiler) dxcCompiler ->Release();
+
+ return 0;
+ }
+
+ String dissassembleDXILUsingDXC(
+ CompileRequest* compileRequest,
+ void const* data,
+ size_t size)
+ {
+ // First deal with all the rigamarole of loading
+ // the `dxcompiler` library, and creating the
+ // top-level COM objects that will be used to
+ // compile things.
+
+ static DxcCreateInstanceProc dxcCreateInstance = nullptr;
+ if (!dxcCreateInstance)
+ {
+ auto dxcSharedLibrary = getDXCSharedLibrary(compileRequest);
+ if (!dxcSharedLibrary)
+ return 1;
+
+ dxcCreateInstance = (DxcCreateInstanceProc) dxcSharedLibrary.findFuncByName("DxcCreateInstance");
+ if (!dxcCreateInstance)
+ return 1;
+ }
+
+ IDxcCompiler* dxcCompiler = nullptr;
+ if (FAILED(dxcCreateInstance(
+ CLSID_DxcCompiler,
+ __uuidof(dxcCompiler),
+ (LPVOID*) &dxcCompiler)))
+ {
+ return 1;
+ }
+
+ IDxcLibrary* dxcLibrary = nullptr;
+ if (FAILED(dxcCreateInstance(
+ CLSID_DxcLibrary,
+ __uuidof(dxcLibrary),
+ (LPVOID*) &dxcLibrary)))
+ {
+ return 1;
+ }
+
+ // Create blob from the input data
+ IDxcBlobEncoding* dxcSourceBlob = nullptr;
+ if (FAILED(dxcLibrary->CreateBlobWithEncodingFromPinned(
+ (LPBYTE) data,
+ (UINT32) size,
+ 0,
+ &dxcSourceBlob)))
+ {
+ return 1;
+ }
+
+ IDxcBlobEncoding* dxcResultBlob = nullptr;
+ if(FAILED(dxcCompiler->Disassemble(
+ dxcSourceBlob,
+ &dxcResultBlob)))
+ {
+ return 1;
+ }
+
+ String result;
+ char const* codeBegin = (char const*)dxcResultBlob->GetBufferPointer();
+ char const* codeEnd = codeBegin + dxcResultBlob->GetBufferSize() - 1;
+ result.append(codeBegin, codeEnd);
+
+ if(dxcResultBlob) dxcResultBlob ->Release();
+ if(dxcLibrary) dxcLibrary ->Release();
+ if(dxcCompiler) dxcCompiler ->Release();
+
+ return result;
+ }
+
+
+} // namespace Slang
+
+#endif
+
+
+
diff --git a/source/slang/options.cpp b/source/slang/options.cpp
index 971d17b51..7eea3fecc 100644
--- a/source/slang/options.cpp
+++ b/source/slang/options.cpp
@@ -317,6 +317,8 @@ struct OptionsParser
CASE(spirv, SPIRV);
CASE(spirv-assembly, SPIRV_ASM);
+ CASE(dxil, DXIL);
+ CASE(dxil-assembly, DXIL_ASM);
CASE(none, TARGET_NONE);
#undef CASE
@@ -390,6 +392,7 @@ struct OptionsParser
String name = tryReadCommandLineArgument(arg, &argCursor, argEnd);
SlangPassThrough passThrough = SLANG_PASS_THROUGH_NONE;
if (name == "fxc") { passThrough = SLANG_PASS_THROUGH_FXC; }
+ else if (name == "dxc") { passThrough = SLANG_PASS_THROUGH_DXC; }
else if (name == "glslang") { passThrough = SLANG_PASS_THROUGH_GLSLANG; }
else
{
diff --git a/source/slang/profile-defs.h b/source/slang/profile-defs.h
index 84153ee46..ea23398e1 100644
--- a/source/slang/profile-defs.h
+++ b/source/slang/profile-defs.h
@@ -77,6 +77,9 @@ PROFILE_VERSION(DX_4_0_Level_9_1, DX)
PROFILE_VERSION(DX_4_0_Level_9_3, DX)
PROFILE_VERSION(DX_4_1, DX)
PROFILE_VERSION(DX_5_0, DX)
+PROFILE_VERSION(DX_6_0, DX)
+PROFILE_VERSION(DX_6_1, DX)
+PROFILE_VERSION(DX_6_2, DX)
PROFILE_VERSION(GLSL_110, GLSL)
PROFILE_VERSION(GLSL_120, GLSL)
@@ -97,23 +100,49 @@ PROFILE_VERSION(GLSL_450, GLSL)
PROFILE(DX_Compute_4_0, cs_4_0, Compute, DX_4_0)
PROFILE(DX_Compute_4_1, cs_4_1, Compute, DX_4_1)
PROFILE(DX_Compute_5_0, cs_5_0, Compute, DX_5_0)
+PROFILE(DX_Compute_6_0, cs_6_0, Compute, DX_6_0)
+PROFILE(DX_Compute_6_1, cs_6_1, Compute, DX_6_1)
+PROFILE(DX_Compute_6_2, cs_6_2, Compute, DX_6_2)
+
PROFILE(DX_Domain_5_0, ds_5_0, Domain, DX_5_0)
+PROFILE(DX_Domain_6_0, ds_6_0, Domain, DX_6_0)
+PROFILE(DX_Domain_6_1, ds_6_1, Domain, DX_6_1)
+PROFILE(DX_Domain_6_2, ds_6_2, Domain, DX_6_2)
+
PROFILE(DX_Geometry_4_0, gs_4_0, Geometry, DX_4_0)
PROFILE(DX_Geometry_4_1, gs_4_1, Geometry, DX_4_1)
PROFILE(DX_Geometry_5_0, gs_5_0, Geometry, DX_5_0)
+PROFILE(DX_Geometry_6_0, gs_6_0, Geometry, DX_6_0)
+PROFILE(DX_Geometry_6_1, gs_6_1, Geometry, DX_6_1)
+PROFILE(DX_Geometry_6_2, gs_6_2, Geometry, DX_6_2)
+
+
PROFILE(DX_Hull_5_0, hs_5_0, Hull, DX_5_0)
+PROFILE(DX_Hull_6_0, hs_6_0, Hull, DX_6_0)
+PROFILE(DX_Hull_6_1, hs_6_1, Hull, DX_6_1)
+PROFILE(DX_Hull_6_2, hs_6_2, Hull, DX_6_2)
+
+
PROFILE(DX_Fragment_4_0, ps_4_0, Fragment, DX_4_0)
PROFILE(DX_Fragment_4_0_Level_9_0, ps_4_0_level_9_0, Fragment, DX_4_0_Level_9_0)
PROFILE(DX_Fragment_4_0_Level_9_1, ps_4_0_level_9_1, Fragment, DX_4_0_Level_9_1)
PROFILE(DX_Fragment_4_0_Level_9_3, ps_4_0_level_9_3, Fragment, DX_4_0_Level_9_3)
PROFILE(DX_Fragment_4_1, ps_4_1, Fragment, DX_4_1)
PROFILE(DX_Fragment_5_0, ps_5_0, Fragment, DX_5_0)
+PROFILE(DX_Fragment_6_0, ps_6_0, Fragment, DX_6_0)
+PROFILE(DX_Fragment_6_1, ps_6_1, Fragment, DX_6_1)
+PROFILE(DX_Fragment_6_2, ps_6_2, Fragment, DX_6_2)
+
+
PROFILE(DX_Vertex_4_0, vs_4_0, Vertex, DX_4_0)
PROFILE(DX_Vertex_4_0_Level_9_0, vs_4_0_level_9_0, Vertex, DX_4_0_Level_9_0)
PROFILE(DX_Vertex_4_0_Level_9_1, vs_4_0_level_9_1, Vertex, DX_4_0_Level_9_1)
PROFILE(DX_Vertex_4_0_Level_9_3, vs_4_0_level_9_3, Vertex, DX_4_0_Level_9_3)
PROFILE(DX_Vertex_4_1, vs_4_1, Vertex, DX_4_1)
PROFILE(DX_Vertex_5_0, vs_5_0, Vertex, DX_5_0)
+PROFILE(DX_Vertex_6_0, vs_6_0, Vertex, DX_6_0)
+PROFILE(DX_Vertex_6_1, vs_6_1, Vertex, DX_6_1)
+PROFILE(DX_Vertex_6_2, vs_6_2, Vertex, DX_6_2)
// Define all the GLSL profiles
diff --git a/source/slang/slang.vcxproj b/source/slang/slang.vcxproj
index 072556cb9..b09eae3ab 100644
--- a/source/slang/slang.vcxproj
+++ b/source/slang/slang.vcxproj
@@ -209,6 +209,7 @@
<ClCompile Include="check.cpp" />
<ClCompile Include="compiler.cpp" />
<ClCompile Include="diagnostics.cpp" />
+ <ClCompile Include="dxc-support.cpp" />
<ClCompile Include="emit.cpp" />
<ClCompile Include="ir.cpp" />
<ClCompile Include="lexer.cpp" />
diff --git a/source/slang/slang.vcxproj.filters b/source/slang/slang.vcxproj.filters
index ba16ac8d8..ef9b3c8e5 100644
--- a/source/slang/slang.vcxproj.filters
+++ b/source/slang/slang.vcxproj.filters
@@ -70,6 +70,7 @@
<ClCompile Include="bytecode.cpp" />
<ClCompile Include="vm.cpp" />
<ClCompile Include="mangle.cpp" />
+ <ClCompile Include="dxc-support.cpp" />
</ItemGroup>
<ItemGroup>
<CustomBuild Include="core.meta.slang" />
diff --git a/source/slang/type-layout.cpp b/source/slang/type-layout.cpp
index 19b9a435e..0825da6b4 100644
--- a/source/slang/type-layout.cpp
+++ b/source/slang/type-layout.cpp
@@ -621,6 +621,8 @@ LayoutRulesFamilyImpl* GetLayoutRulesFamilyImpl(CodeGenTarget target)
case CodeGenTarget::HLSL:
case CodeGenTarget::DXBytecode:
case CodeGenTarget::DXBytecodeAssembly:
+ case CodeGenTarget::DXIL:
+ case CodeGenTarget::DXILAssembly:
return &kHLSLLayoutRulesFamilyImpl;
case CodeGenTarget::GLSL: