1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
|
// unit-test-translation-unit-import.cpp
#include "slang.h"
#include <stdio.h>
#include <stdlib.h>
#include "tools/unit-test/slang-unit-test.h"
#include "slang-com-ptr.h"
#include "../../source/core/slang-io.h"
#include "../../source/core/slang-process.h"
using namespace Slang;
static String getTypeFullName(slang::TypeReflection* type)
{
ComPtr<ISlangBlob> blob;
type->getFullName(blob.writeRef());
return String((const char*)blob->getBufferPointer());
}
// Test that the reflection API provides correct info about entry point and ordinary functions.
SLANG_UNIT_TEST(functionReflection)
{
// Source for a module that contains an undecorated entrypoint.
const char* userSourceBody = R"(
[__AttributeUsage(_AttributeTargets.Function)]
struct MyFuncPropertyAttribute {int v;}
[MyFuncProperty(1024)]
[Differentiable]
float ordinaryFunc(no_diff float x, int y) { return x + y; }
float4 fragMain(float4 pos:SV_Position) : SV_Position
{
return pos;
}
float foo(float x) { return x; }
float foo(float x, uint i) { return x + i; }
)";
auto moduleName = "moduleG" + String(Process::getId());
String userSource = "import " + moduleName + ";\n" + userSourceBody;
ComPtr<slang::IGlobalSession> globalSession;
SLANG_CHECK(slang_createGlobalSession(SLANG_API_VERSION, globalSession.writeRef()) == SLANG_OK);
slang::TargetDesc targetDesc = {};
targetDesc.format = SLANG_HLSL;
targetDesc.profile = globalSession->findProfile("sm_5_0");
slang::SessionDesc sessionDesc = {};
sessionDesc.targetCount = 1;
sessionDesc.targets = &targetDesc;
ComPtr<slang::ISession> session;
SLANG_CHECK(globalSession->createSession(sessionDesc, session.writeRef()) == SLANG_OK);
ComPtr<slang::IBlob> diagnosticBlob;
auto module = session->loadModuleFromSourceString("m", "m.slang", userSourceBody, diagnosticBlob.writeRef());
SLANG_CHECK(module != nullptr);
ComPtr<slang::IEntryPoint> entryPoint;
module->findAndCheckEntryPoint("fragMain", SLANG_STAGE_FRAGMENT, entryPoint.writeRef(), diagnosticBlob.writeRef());
SLANG_CHECK(entryPoint != nullptr);
auto entryPointFuncReflection = entryPoint->getFunctionReflection();
SLANG_CHECK(entryPointFuncReflection != nullptr);
SLANG_CHECK(UnownedStringSlice(entryPointFuncReflection->getName()) == "fragMain");
SLANG_CHECK(entryPointFuncReflection->getParameterCount() == 1);
SLANG_CHECK(UnownedStringSlice(entryPointFuncReflection->getParameterByIndex(0)->getName()) == "pos");
SLANG_CHECK(getTypeFullName(entryPointFuncReflection->getParameterByIndex(0)->getType()) == "vector<float,4>");
auto funcReflection = module->getLayout()->findFunctionByName("ordinaryFunc");
SLANG_CHECK(funcReflection != nullptr);
SLANG_CHECK(funcReflection->findModifier(slang::Modifier::Differentiable) != nullptr);
SLANG_CHECK(getTypeFullName(funcReflection->getReturnType()) == "float");
SLANG_CHECK(UnownedStringSlice(funcReflection->getName()) == "ordinaryFunc");
SLANG_CHECK(funcReflection->getParameterCount() == 2);
SLANG_CHECK(UnownedStringSlice(funcReflection->getParameterByIndex(0)->getName()) == "x");
SLANG_CHECK(getTypeFullName(funcReflection->getParameterByIndex(0)->getType()) == "float");
SLANG_CHECK(funcReflection->getParameterByIndex(0)->findModifier(slang::Modifier::NoDiff) != nullptr);
SLANG_CHECK(UnownedStringSlice(funcReflection->getParameterByIndex(1)->getName()) == "y");
SLANG_CHECK(getTypeFullName(funcReflection->getParameterByIndex(1)->getType()) == "int");
SLANG_CHECK(funcReflection->getUserAttributeCount() == 1);
auto userAttribute = funcReflection->getUserAttributeByIndex(0);
SLANG_CHECK(UnownedStringSlice(userAttribute->getName()) == "MyFuncProperty");
SLANG_CHECK(userAttribute->getArgumentCount() == 1);
SLANG_CHECK(getTypeFullName(userAttribute->getArgumentType(0)) == "int");
int val = 0;
auto result = userAttribute->getArgumentValueInt(0, &val);
SLANG_CHECK(result == SLANG_OK);
SLANG_CHECK(val == 1024);
SLANG_CHECK(funcReflection->findUserAttributeByName(globalSession.get(), "MyFuncProperty") == userAttribute);
// Check overloaded method resolution
auto overloadReflection = module->getLayout()->findFunctionByName("foo");
SLANG_CHECK(overloadReflection != nullptr);
SLANG_CHECK(overloadReflection->isOverloaded() == true);
SLANG_CHECK(overloadReflection->getOverloadCount() == 2);
auto firstOverload = overloadReflection->getOverload(0);
SLANG_CHECK(firstOverload != nullptr);
SLANG_CHECK(UnownedStringSlice(firstOverload->getName()) == "foo");
SLANG_CHECK(firstOverload->getParameterCount() == 2);
SLANG_CHECK(UnownedStringSlice(firstOverload->getParameterByIndex(0)->getName()) == "x");
SLANG_CHECK(getTypeFullName(firstOverload->getParameterByIndex(0)->getType()) == "float");
SLANG_CHECK(UnownedStringSlice(firstOverload->getParameterByIndex(1)->getName()) == "i");
SLANG_CHECK(getTypeFullName(firstOverload->getParameterByIndex(1)->getType()) == "uint");
auto secondOverload = overloadReflection->getOverload(1);
SLANG_CHECK(secondOverload != nullptr);
SLANG_CHECK(UnownedStringSlice(secondOverload->getName()) == "foo");
SLANG_CHECK(secondOverload->getParameterCount() == 1);
SLANG_CHECK(UnownedStringSlice(secondOverload->getParameterByIndex(0)->getName()) == "x");
// Check overload resolution via argument types.
slang::TypeReflection* argTypes[] = {
module->getLayout()->findTypeByName("float"),
module->getLayout()->findTypeByName("uint"),
};
auto resolvedFunctionReflection = overloadReflection->specializeWithArgTypes(2, argTypes);
SLANG_CHECK(resolvedFunctionReflection == firstOverload);
}
|