Skip to content

Commit 42f4993

Browse files
authored
Add gfx interface definition in Slang. (shader-slang#2364)
* Add gfx interface definition in Slang. - add gfx interface definitons in Slang. - fix slang compiler to correctly type-check `out` interface argument. - modify gfx interface to be fully COM compatible - add convenient ShaderProgram creation methods to gfx. * Fix compile errors and warnings. * Update project files * Fix cuda. * Properly implement queryInterface in command encoder impls. Co-authored-by: Yong He <yhe@nvidia.com>
1 parent e68fab2 commit 42f4993

33 files changed

+3098
-159
lines changed

build/visual-studio/gfx/gfx.vcxproj

+4
Original file line numberDiff line numberDiff line change
@@ -538,6 +538,10 @@
538538
<ClCompile Include="..\..\..\tools\gfx\vulkan\vk-transient-heap.cpp" />
539539
<ClCompile Include="..\..\..\tools\gfx\vulkan\vk-util.cpp" />
540540
</ItemGroup>
541+
<ItemGroup>
542+
<None Include="..\..\..\tools\gfx\gfx.slang" />
543+
<None Include="..\..\..\tools\gfx\slang.slang" />
544+
</ItemGroup>
541545
<ItemGroup>
542546
<ProjectReference Include="..\core\core.vcxproj">
543547
<Project>{F9BE7957-8399-899E-0C49-E714FDDD4B65}</Project>

build/visual-studio/gfx/gfx.vcxproj.filters

+8
Original file line numberDiff line numberDiff line change
@@ -720,4 +720,12 @@
720720
<Filter>Source Files</Filter>
721721
</ClCompile>
722722
</ItemGroup>
723+
<ItemGroup>
724+
<None Include="..\..\..\tools\gfx\gfx.slang">
725+
<Filter>Source Files</Filter>
726+
</None>
727+
<None Include="..\..\..\tools\gfx\slang.slang">
728+
<Filter>Source Files</Filter>
729+
</None>
730+
</ItemGroup>
723731
</Project>

examples/ray-tracing/main.cpp

+2-2
Original file line numberDiff line numberDiff line change
@@ -404,13 +404,13 @@ Slang::Result initialize()
404404
IBufferResource::Desc asBufferDesc;
405405
asBufferDesc.type = IResource::Type::Buffer;
406406
asBufferDesc.defaultState = ResourceState::AccelerationStructure;
407-
asBufferDesc.sizeInBytes = compactedSize;
407+
asBufferDesc.sizeInBytes = (Size)compactedSize;
408408
gBLASBuffer = gDevice->createBufferResource(asBufferDesc);
409409
IAccelerationStructure::CreateDesc createDesc;
410410
createDesc.buffer = gBLASBuffer;
411411
createDesc.kind = IAccelerationStructure::Kind::BottomLevel;
412412
createDesc.offset = 0;
413-
createDesc.size = compactedSize;
413+
createDesc.size = (Size)compactedSize;
414414
gDevice->createAccelerationStructure(createDesc, gBLAS.writeRef());
415415

416416
commandBuffer = gTransientHeaps[0]->createCommandBuffer();

prelude/slang-cpp-host-prelude.h

+2
Original file line numberDiff line numberDiff line change
@@ -5,6 +5,8 @@
55
#include <cmath>
66
#include <cstring>
77

8+
#define SLANG_COM_PTR_ENABLE_REF_OPERATOR 1
9+
810
#include "../source/slang-rt/slang-rt.h"
911
#include "../slang-com-ptr.h"
1012
#include "slang-cpp-types.h"

slang-com-ptr.h

+2
Original file line numberDiff line numberDiff line change
@@ -111,7 +111,9 @@ class ComPtr
111111
protected:
112112
/// Gets the address of the dumb pointer.
113113
// Disabled: use writeRef and readRef to get a reference based on usage.
114+
#ifndef SLANG_COM_PTR_ENABLE_REF_OPERATOR
114115
SLANG_FORCE_INLINE T** operator&() = delete;
116+
#endif
115117

116118
T* m_ptr;
117119
};

slang-gfx.h

+40-5
Original file line numberDiff line numberDiff line change
@@ -130,6 +130,14 @@ const GfxCount kMaxRenderTargetCount = 8;
130130

131131
class ITransientResourceHeap;
132132

133+
enum class ShaderModuleSourceType
134+
{
135+
SlangSource, // a slang source string in memory.
136+
SlangModuleBinary, // a slang module binary code in memory.
137+
SlangSourceFile, // a slang source from file.
138+
SlangModuleBinaryFile, // a slang module binary code from file.
139+
};
140+
133141
class IShaderProgram: public ISlangUnknown
134142
{
135143
public:
@@ -162,6 +170,22 @@ class IShaderProgram: public ISlangUnknown
162170
// Each element must define only 1 Slang EntryPoint.
163171
slang::IComponentType** slangEntryPoints = nullptr;
164172
};
173+
174+
struct CreateDesc2
175+
{
176+
ShaderModuleSourceType sourceType;
177+
void* sourceData;
178+
Size sourceDataSize;
179+
180+
// Number of entry points to include in the shader program. 0 means include all entry points
181+
// defined in the module.
182+
GfxCount entryPointCount = 0;
183+
// Names of entry points to include in the shader program. The size of the array must be
184+
// `entryPointCount`.
185+
const char** entryPointNames = nullptr;
186+
};
187+
188+
virtual SLANG_NO_THROW slang::TypeReflection* SLANG_MCALL findTypeByName(const char* name) = 0;
165189
};
166190
#define SLANG_UUID_IShaderProgram \
167191
{ \
@@ -1516,8 +1540,9 @@ class IQueryPool : public ISlangUnknown
15161540
{ 0xc2cc3784, 0x12da, 0x480a, { 0xa8, 0x74, 0x8b, 0x31, 0x96, 0x1c, 0xa4, 0x36 } }
15171541

15181542

1519-
class ICommandEncoder
1543+
class ICommandEncoder : public ISlangUnknown
15201544
{
1545+
SLANG_COM_INTERFACE( 0x77ea6383, 0xbe3d, 0x40aa, { 0x8b, 0x45, 0xfd, 0xf0, 0xd7, 0x5b, 0xfa, 0x34 });
15211546
public:
15221547
virtual SLANG_NO_THROW void SLANG_MCALL endEncoding() = 0;
15231548
virtual SLANG_NO_THROW void SLANG_MCALL writeTimestamp(IQueryPool* queryPool, GfxIndex queryIndex) = 0;
@@ -1566,6 +1591,9 @@ struct ClearResourceViewFlags
15661591

15671592
class IResourceCommandEncoder : public ICommandEncoder
15681593
{
1594+
// {F99A00E9-ED50-4088-8A0E-3B26755031EA}
1595+
SLANG_COM_INTERFACE(0xf99a00e9, 0xed50, 0x4088, { 0x8a, 0xe, 0x3b, 0x26, 0x75, 0x50, 0x31, 0xea });
1596+
15691597
public:
15701598
virtual SLANG_NO_THROW void SLANG_MCALL copyBuffer(
15711599
IBufferResource* dst,
@@ -1646,6 +1674,8 @@ class IResourceCommandEncoder : public ICommandEncoder
16461674

16471675
class IRenderCommandEncoder : public IResourceCommandEncoder
16481676
{
1677+
// {7A8D56D0-53E6-4AD6-85F7-D14DC110FDCE}
1678+
SLANG_COM_INTERFACE(0x7a8d56d0, 0x53e6, 0x4ad6, { 0x85, 0xf7, 0xd1, 0x4d, 0xc1, 0x10, 0xfd, 0xce })
16491679
public:
16501680
// Sets the current pipeline state. This method returns a transient shader object for
16511681
// writing shader parameters. This shader object will not retain any resources or
@@ -1728,6 +1758,9 @@ class IRenderCommandEncoder : public IResourceCommandEncoder
17281758

17291759
class IComputeCommandEncoder : public IResourceCommandEncoder
17301760
{
1761+
// {88AA9322-82F7-4FE6-A68A-29C7FE798737}
1762+
SLANG_COM_INTERFACE(0x88aa9322, 0x82f7, 0x4fe6, { 0xa6, 0x8a, 0x29, 0xc7, 0xfe, 0x79, 0x87, 0x37 })
1763+
17311764
public:
17321765
// Sets the current pipeline state. This method returns a transient shader object for
17331766
// writing shader parameters. This shader object will not retain any resources or
@@ -1765,6 +1798,7 @@ struct AccelerationStructureQueryDesc
17651798

17661799
class IRayTracingCommandEncoder : public IResourceCommandEncoder
17671800
{
1801+
SLANG_COM_INTERFACE(0x9a672b87, 0x5035, 0x45e3, { 0x96, 0x7c, 0x1f, 0x85, 0xcd, 0xb3, 0x63, 0x4f })
17681802
public:
17691803
virtual SLANG_NO_THROW void SLANG_MCALL buildAccelerationStructure(
17701804
const IAccelerationStructure::BuildDesc& desc,
@@ -1799,10 +1833,6 @@ class IRayTracingCommandEncoder : public IResourceCommandEncoder
17991833
GfxCount height,
18001834
GfxCount depth) = 0;
18011835
};
1802-
#define SLANG_UUID_IRayTracingCommandEncoder \
1803-
{ \
1804-
0x9a672b87, 0x5035, 0x45e3, { 0x96, 0x7c, 0x1f, 0x85, 0xcd, 0xb3, 0x63, 0x4f } \
1805-
}
18061836

18071837
class ICommandBuffer : public ISlangUnknown
18081838
{
@@ -2356,6 +2386,11 @@ class IDevice: public ISlangUnknown
23562386
return program;
23572387
}
23582388

2389+
virtual SLANG_NO_THROW Result SLANG_MCALL createProgram2(
2390+
const IShaderProgram::CreateDesc2& createDesc,
2391+
IShaderProgram** outProgram,
2392+
ISlangBlob** outDiagnosticBlob = nullptr) = 0;
2393+
23592394
virtual SLANG_NO_THROW Result SLANG_MCALL createGraphicsPipelineState(
23602395
const GraphicsPipelineStateDesc& desc,
23612396
IPipelineState** outState) = 0;

slang.h

+9
Original file line numberDiff line numberDiff line change
@@ -4449,6 +4449,15 @@ namespace slang
44494449
virtual SLANG_NO_THROW SlangResult SLANG_MCALL findEntryPointByName(
44504450
char const* name,
44514451
IEntryPoint** outEntryPoint) = 0;
4452+
4453+
/// Get number of entry points defined in the module. An entry point defined in a module
4454+
/// is by default not included in the linkage, so calls to `IComponentType::getEntryPointCount`
4455+
/// on an `IModule` instance will always return 0. However `IModule::getDefinedEntryPointCount`
4456+
/// will return the number of defined entry points.
4457+
virtual SLANG_NO_THROW SlangInt32 SLANG_MCALL getDefinedEntryPointCount() = 0;
4458+
/// Get the name of an entry point defined in the module.
4459+
virtual SLANG_NO_THROW SlangResult SLANG_MCALL
4460+
getDefinedEntryPoint(SlangInt32 index, IEntryPoint** outEntryPoint) = 0;
44524461
};
44534462

44544463
#define SLANG_UUID_IModule IModule::getTypeGuid()

source/slang/core.meta.slang

+77
Original file line numberDiff line numberDiff line change
@@ -428,20 +428,29 @@ extension bool
428428
__implicit_conversion($(kConversionCost_PtrToBool))
429429
__intrinsic_op($(kIROp_CastPtrToBool))
430430
__init(Ptr<T> ptr);
431+
432+
static bool maxValue = true;
433+
static bool minValue = false;
431434
}
432435

433436
extension uint64_t
434437
{
435438
__generic<T>
436439
__intrinsic_op($(kIROp_Construct))
437440
__init(Ptr<T> ptr);
441+
442+
static uint64_t maxValue = 0xFFFFFFFFFFFFFFFFULL;
443+
static uint64_t minValue = 0;
438444
}
439445

440446
extension int64_t
441447
{
442448
__generic<T>
443449
__intrinsic_op($(kIROp_Construct))
444450
__init(Ptr<T> ptr);
451+
452+
static int64_t maxValue = 0x7FFFFFFFFFFFFFFFLL;
453+
static int64_t minValue = -0x8000000000000000LL;
445454
}
446455

447456
__generic<T>
@@ -509,6 +518,28 @@ bool operator!=(__none_t noneVal, Optional<T> val)
509518
return val.hasValue;
510519
}
511520

521+
__generic<T>
522+
__magic_type(NativeRefType)
523+
__intrinsic_type($(kIROp_NativePtrType))
524+
struct NativeRef
525+
{
526+
__intrinsic_op($(kIROp_GetNativePtr))
527+
__init(T val);
528+
};
529+
530+
__generic<T>
531+
__intrinsic_op($(kIROp_ManagedPtrAttach))
532+
void __managed_ptr_attach(__ref T val, NativeRef<T> nativeVal);
533+
534+
__generic<T>
535+
[__unsafeForceInlineEarly]
536+
T __attachToNativeRef(NativeRef<T> nativeVal)
537+
{
538+
T result;
539+
__managed_ptr_attach(result, nativeVal);
540+
return result;
541+
}
542+
512543
__magic_type(StringType)
513544
__intrinsic_type($(kIROp_StringType))
514545
struct String
@@ -524,6 +555,52 @@ __intrinsic_type($(kIROp_DynamicType))
524555
struct __Dynamic
525556
{};
526557

558+
extension float
559+
{
560+
static float maxValue = 340282346638528859811704183484516925440.0f;
561+
}
562+
563+
extension double
564+
{
565+
static double maxValue = 179769313486231570814527423731704356798070567525844996598917476803157260780028538760589558632766878171540458953514382464234321326889464182768467546703537516986049910576551282076245490090389328944075868508455133942304583236903222948165808559332123348274797826204144723168738177180919299881250404026184124858368.0;
566+
}
567+
568+
extension int
569+
{
570+
static int maxValue = 2147483647;
571+
static int minValue = -2147483648;
572+
}
573+
574+
extension uint
575+
{
576+
static uint maxValue = 4294967295;
577+
static uint minValue = 0;
578+
}
579+
580+
extension int8_t
581+
{
582+
static int8_t maxValue = 127;
583+
static int8_t minValue = -128;
584+
}
585+
586+
extension uint16_t
587+
{
588+
static uint16_t maxValue = 255;
589+
static uint16_t minValue = 0;
590+
}
591+
592+
extension int16_t
593+
{
594+
static int16_t maxValue = 32767;
595+
static int16_t minValue = -32768;
596+
}
597+
598+
extension uint16_t
599+
{
600+
static uint16_t maxValue = 65535;
601+
static uint16_t minValue = 0;
602+
}
603+
527604
/// An `N` component vector with elements of type `T`.
528605
__generic<T = float, let N : int = 4>
529606
__magic_type(Vector)

source/slang/slang-ast-type.h

+7
Original file line numberDiff line numberDiff line change
@@ -583,6 +583,13 @@ class OptionalType : public BuiltinType
583583
Type* getValueType();
584584
};
585585

586+
// A raw-pointer reference to an managed value.
587+
class NativeRefType : public BuiltinType
588+
{
589+
SLANG_AST_CLASS(NativeRefType)
590+
Type* getValueType();
591+
};
592+
586593
// A type alias of some kind (e.g., via `typedef`)
587594
class NamedExpressionType : public Type
588595
{

source/slang/slang-check-decl.cpp

+29-1
Original file line numberDiff line numberDiff line change
@@ -3063,10 +3063,37 @@ namespace Slang
30633063
InheritanceDecl* inheritanceDecl,
30643064
ContainerDecl* parentDecl)
30653065
{
3066+
auto superType = inheritanceDecl->base.type;
3067+
30663068
if( auto declRefType = as<DeclRefType>(subType) )
30673069
{
30683070
auto declRef = declRefType->declRef;
30693071

3072+
if (auto superDeclRefType = as<DeclRefType>(superType))
3073+
{
3074+
auto superTypeDecl = superDeclRefType->declRef.getDecl();
3075+
if (superTypeDecl->findModifier<ComInterfaceAttribute>())
3076+
{
3077+
// A struct cannot implement a COM Interface.
3078+
if (auto classDecl = as<ClassDecl>(superTypeDecl))
3079+
{
3080+
// OK.
3081+
SLANG_UNUSED(classDecl);
3082+
}
3083+
else if (auto subInterfaceDecl = as<InterfaceDecl>(superTypeDecl))
3084+
{
3085+
if (!subInterfaceDecl->findModifier<ComInterfaceAttribute>())
3086+
{
3087+
getSink()->diagnose(inheritanceDecl, Diagnostics::interfaceInheritingComMustBeCom);
3088+
}
3089+
}
3090+
else if (auto structDecl = as<StructDecl>(superTypeDecl))
3091+
{
3092+
getSink()->diagnose(inheritanceDecl, Diagnostics::structCannotImplementComInterface);
3093+
}
3094+
}
3095+
}
3096+
30703097
// Don't check conformances for abstract types that
30713098
// are being used to express *required* conformances.
30723099
if (auto assocTypeDeclRef = declRef.as<AssocTypeDecl>())
@@ -3089,11 +3116,12 @@ namespace Slang
30893116
// code to work.
30903117
return true;
30913118
}
3119+
3120+
30923121
}
30933122

30943123
// Look at the type being inherited from, and validate
30953124
// appropriately.
3096-
auto superType = inheritanceDecl->base.type;
30973125

30983126
DeclaredSubtypeWitness* subIsSuperWitness = m_astBuilder->create<DeclaredSubtypeWitness>();
30993127
subIsSuperWitness->declRef = makeDeclRef(inheritanceDecl);

source/slang/slang-check-overload.cpp

+8-1
Original file line numberDiff line numberDiff line change
@@ -1443,8 +1443,15 @@ namespace Slang
14431443
arg = maybeOpenRef(arg);
14441444
}
14451445

1446-
for (auto& arg : expr->arguments)
1446+
auto funcType = as<FuncType>(funcExprType);
1447+
for (Index i = 0; i < expr->arguments.getCount(); i++)
14471448
{
1449+
auto& arg = expr->arguments[i];
1450+
if (funcType && i < (Index)funcType->getParamCount())
1451+
{
1452+
if (funcType->getParamDirection(i) == kParameterDirection_Out)
1453+
continue;
1454+
}
14481455
arg = maybeOpenExistential(arg);
14491456
}
14501457

0 commit comments

Comments
 (0)