diff options
| author | Yong He <yonghe@outlook.com> | 2020-06-05 19:38:43 -0700 |
|---|---|---|
| committer | Yong He <yonghe@outlook.com> | 2020-06-05 19:43:30 -0700 |
| commit | 956ede9be7d7b798ce719a7ac6a3be3f472f6db7 (patch) | |
| tree | 8972b17c7f66e17e6f1f639277065aa983cf16f1 | |
| parent | 7d4432ba5ca007b18dfa6e0c19b4598e42c1e505 (diff) | |
Filter lookup results from interfaces in `visitMemberExpr`.
Fixes #1377
| -rw-r--r-- | source/slang/slang-check-expr.cpp | 30 | ||||
| -rw-r--r-- | tests/compute/assoctype-lookup.slang | 58 | ||||
| -rw-r--r-- | tests/compute/assoctype-lookup.slang.expected.txt | 4 |
3 files changed, 91 insertions, 1 deletions
diff --git a/source/slang/slang-check-expr.cpp b/source/slang/slang-check-expr.cpp index 307ec6316..7adda8c77 100644 --- a/source/slang/slang-check-expr.cpp +++ b/source/slang/slang-check-expr.cpp @@ -1826,10 +1826,38 @@ namespace Slang // expr->baseExpression = maybeOpenExistential(expr->baseExpression); - // TODO: Handle the case of an overloaded base expression + // Handle the case of an overloaded base expression // here, in case we can use the name of the member to // disambiguate which of the candidates is meant, or if // we can return an overloaded result. + if (auto overloadedExpr = as<OverloadedExpr>(expr->baseExpression)) + { + if (overloadedExpr->base) + { + // If a member (dynamic or static) lookup result contains both the actual definition + // and the interface definition obtained from inheritance, we want to filter out + // the interface definitions. + LookupResult filteredLookupResult; + for (auto lookupResult : overloadedExpr->lookupResult2) + { + bool shouldRemove = false; + if (lookupResult.declRef.getParent().as<InterfaceDecl>()) + shouldRemove = true; + if (!shouldRemove) + { + filteredLookupResult.items.add(lookupResult); + } + } + if (filteredLookupResult.items.getCount() == 1) + filteredLookupResult.item = filteredLookupResult.items.getFirst(); + expr->baseExpression = createLookupResultExpr( + expr->name, + filteredLookupResult, + overloadedExpr->base, + overloadedExpr->loc); + } + // TODO: handle other cases of OverloadedExpr that need filtering. + } auto & baseType = expr->baseExpression->type; diff --git a/tests/compute/assoctype-lookup.slang b/tests/compute/assoctype-lookup.slang new file mode 100644 index 000000000..1dccc8e12 --- /dev/null +++ b/tests/compute/assoctype-lookup.slang @@ -0,0 +1,58 @@ +// assoctype-nested.slang
+
+// Confirm that an associated type can be correctly looked up.
+
+//TEST(compute):COMPARE_COMPUTE_EX:-slang -compute -cpu
+//TEST(compute):COMPARE_COMPUTE_EX:-slang -compute
+//TEST(compute):COMPARE_COMPUTE_EX:-slang -compute -dx12
+//TEST(compute, vulkan):COMPARE_COMPUTE_EX:-vk -compute
+
+interface IBoneWeightSet
+{
+ associatedtype PackedType;
+};
+
+struct StandardBoneWeightSet : IBoneWeightSet
+{
+ struct PackedType
+ {
+ uint boneIds : BONEIDS;
+ uint boneWeights : BONEWEIGHTS;
+ };
+ PackedType field;
+};
+
+interface IVertexFormat
+{
+ associatedtype BoneWeightSet : IBoneWeightSet;
+};
+
+struct VertexFormat<TBoneWeightSet : IBoneWeightSet> : IVertexFormat
+{
+ typedef TBoneWeightSet BoneWeightSet;
+ TBoneWeightSet.PackedType weightSet;
+};
+
+struct OutputType
+{
+ VertexFormat<StandardBoneWeightSet>.BoneWeightSet.PackedType field;
+};
+
+int test(int val)
+{
+ OutputType rs;
+ rs.field.boneIds = 256;
+ return rs.field.boneIds;
+}
+
+//TEST_INPUT:ubuffer(data=[0 0 0 0], stride=4):out,name=gOutputBuffer
+RWStructuredBuffer<int> gOutputBuffer;
+
+[numthreads(4, 1, 1)]
+void computeMain(uint3 dispatchThreadID : SV_DispatchThreadID)
+{
+ uint tid = dispatchThreadID.x;
+ int inputVal = tid;
+ int outputVal = test(inputVal);
+ gOutputBuffer[tid] = outputVal;
+}
\ No newline at end of file diff --git a/tests/compute/assoctype-lookup.slang.expected.txt b/tests/compute/assoctype-lookup.slang.expected.txt new file mode 100644 index 000000000..54248c25a --- /dev/null +++ b/tests/compute/assoctype-lookup.slang.expected.txt @@ -0,0 +1,4 @@ +100 +100 +100 +100
\ No newline at end of file |
