From 8e86121b6ce0f2995050ebe112306ded6bc87ca2 Mon Sep 17 00:00:00 2001 From: Yong He Date: Wed, 8 May 2024 20:52:36 -0700 Subject: Support `[__ref]` attribute to make `this` pass by reference. (#4139) Fixes #4110. --- source/slang/slang-check-decl.cpp | 33 ++++++++++++++++++++++++++++++++- 1 file changed, 32 insertions(+), 1 deletion(-) (limited to 'source/slang/slang-check-decl.cpp') diff --git a/source/slang/slang-check-decl.cpp b/source/slang/slang-check-decl.cpp index 0d089874e..875fddf2f 100644 --- a/source/slang/slang-check-decl.cpp +++ b/source/slang/slang-check-decl.cpp @@ -2753,6 +2753,15 @@ namespace Slang return false; } + if (satisfyingMemberDeclRef.getDecl()->hasModifier() + != requiredMemberDeclRef.getDecl()->hasModifier()) + { + // A `[ref]` method can't satisfy a non-`[ref]` requirement. + // The opposite direction is okay, but we will need to synthesize a wrapper + // to ensure type matches, so we will return false here either way. + return false; + } + if(satisfyingMemberDeclRef.getDecl()->hasModifier() != requiredMemberDeclRef.getDecl()->hasModifier()) { @@ -3647,6 +3656,16 @@ namespace Slang auto synConstRefAttr = m_astBuilder->create(); addModifier(synthesized, synConstRefAttr); } + if (requiredMemberDeclRef.getDecl()->hasModifier()) + { + // If the interface requirement is `[ref]` then our + // synthesized method should be too. + // + synThis->type.isLeftValue = true; + + auto synConstRefAttr = m_astBuilder->create(); + addModifier(synthesized, synConstRefAttr); + } if (requiredMemberDeclRef.getDecl()->hasModifier()) { auto noDiffThisAttr = m_astBuilder->create(); @@ -4311,7 +4330,18 @@ namespace Slang auto synAttr = m_astBuilder->create(); synAccessorDecl->modifiers.first = synAttr; } + else if (requiredAccessorDeclRef.getDecl()->hasModifier()) + { + synThis->type.isLeftValue = true; + auto synAttr = m_astBuilder->create(); + synAccessorDecl->modifiers.first = synAttr; + } + else if (requiredAccessorDeclRef.getDecl()->hasModifier()) + { + auto synAttr = m_astBuilder->create(); + synAccessorDecl->modifiers.first = synAttr; + } // We are going to synthesize an expression and then perform // semantic checking on it, but if there are semantic errors // we do *not* want to report them to the user as such, and @@ -7733,7 +7763,8 @@ namespace Slang if (!isEffectivelyStatic(decl)) { auto constrefAttr = decl->findModifier(); - if (constrefAttr) + auto refAttr = decl->findModifier(); + if (constrefAttr || refAttr) { if (isTypeDifferentiable(calcThisType(getParentDecl(decl)))) { -- cgit v1.2.3