summaryrefslogtreecommitdiff
path: root/source/slang/slang-lower-to-ir.cpp
diff options
context:
space:
mode:
authorYong He <yonghe@outlook.com>2021-09-03 01:57:31 -0700
committerGitHub <noreply@github.com>2021-09-03 01:57:31 -0700
commit0a81d11bc06e55089f7061225b9553329f697828 (patch)
tree31cc310eeb7cae68a5e3e49d882578da7e565dde /source/slang/slang-lower-to-ir.cpp
parentcf7dddae74ae990c2b46a9feeaea3b7a33ec077d (diff)
Fix crash: dynamic dispatch of generic interface method. (#1929)
* Fix crash: dynamic dispatch of generic interface method. * Fix memory error. Co-authored-by: Yong He <yhe@nvidia.com>
Diffstat (limited to 'source/slang/slang-lower-to-ir.cpp')
-rw-r--r--source/slang/slang-lower-to-ir.cpp24
1 files changed, 21 insertions, 3 deletions
diff --git a/source/slang/slang-lower-to-ir.cpp b/source/slang/slang-lower-to-ir.cpp
index 8a40dbde9..c1738887d 100644
--- a/source/slang/slang-lower-to-ir.cpp
+++ b/source/slang/slang-lower-to-ir.cpp
@@ -6702,9 +6702,18 @@ struct DeclLoweringVisitor : DeclVisitor<DeclLoweringVisitor, LoweredValInfo>
{
HashSet<IRInst*> valuesToClone;
markInstsToClone(valuesToClone, parentGeneric->getFirstBlock(), returnType);
+ // For Function Types, we always clone all generic parameters regardless of whether
+ // the generic parameter appears in the function signature or not.
+ if (returnType->getOp() == kIROp_FuncType)
+ {
+ for (auto genericParam : parentGeneric->getParams())
+ {
+ markInstsToClone(valuesToClone, parentGeneric->getFirstBlock(), genericParam);
+ }
+ }
if (valuesToClone.Count() == 0)
{
- // If returnType is independent of generic parameters, set
+ // If the new generic has no parameters, set
// the generic inst's type to just `returnType`.
parentGeneric->setFullType((IRType*)returnType);
}
@@ -6727,8 +6736,17 @@ struct DeclLoweringVisitor : DeclVisitor<DeclLoweringVisitor, LoweredValInfo>
}
IRInst* clonedReturnType = nullptr;
cloneEnv.mapOldValToNew.TryGetValue(returnType, clonedReturnType);
- SLANG_ASSERT(clonedReturnType);
- typeBuilder.emitReturn(clonedReturnType);
+ if (clonedReturnType)
+ {
+ // If the type has explicit dependency on generic parameters, use
+ // the cloned type.
+ typeBuilder.emitReturn(clonedReturnType);
+ }
+ else
+ {
+ // Otherwise just use the original type value directly.
+ typeBuilder.emitReturn(returnType);
+ }
parentGeneric->setFullType((IRType*)typeGeneric);
returnType = typeGeneric;
}