summaryrefslogtreecommitdiffstats
path: root/source/slang
diff options
context:
space:
mode:
authorTim Foley <tfoley@nvidia.com>2017-10-30 10:03:52 -0700
committerTim Foley <tfoley@nvidia.com>2017-10-30 10:26:22 -0700
commit11f44241ffef478560eaba79af330c16f0bc8d69 (patch)
tree5f7b06149f5d6e0af9241db8f60b28b9c07785f1 /source/slang
parent42f1cff5c1471e6bc3988a9810c20b8bcc1c84dd (diff)
Allow for implicit `this` expressions.
- When peforming ordinary lookup, if the container declaration for a scope is an aggregate type or `extension` decl, then use a "breadcrumb" to make sure that we use a `this` expression as the base of any resulting declaration reference - Add a test case for implicit `this` usage - Update constrained generic test case to use implicit `this` for member reference, as was originally intended
Diffstat (limited to 'source/slang')
-rw-r--r--source/slang/check.cpp34
-rw-r--r--source/slang/lookup.cpp53
-rw-r--r--source/slang/syntax.h6
3 files changed, 64 insertions, 29 deletions
diff --git a/source/slang/check.cpp b/source/slang/check.cpp
index 9b707d218..0c35c4bf4 100644
--- a/source/slang/check.cpp
+++ b/source/slang/check.cpp
@@ -196,6 +196,16 @@ namespace Slang
return derefExpr;
}
+ RefPtr<Expr> createImplicitThisMemberExpr(
+ Type* type,
+ SourceLoc loc)
+ {
+ RefPtr<ThisExpr> expr = new ThisExpr();
+ expr->type = type;
+ expr->loc = loc;
+ return expr;
+ }
+
RefPtr<Expr> ConstructLookupResultExpr(
LookupResultItem const& item,
RefPtr<Expr> baseExpr,
@@ -228,6 +238,30 @@ namespace Slang
}
break;
+ case LookupResultItem::Breadcrumb::Kind::This:
+ {
+ // We expect a `this` to always come
+ // at the start of a chain.
+ SLANG_ASSERT(bb == nullptr);
+
+ // The member was looked up via a `this` expression,
+ // so we need to create one here.
+ if (auto extensionDeclRef = breadcrumb->declRef.As<ExtensionDecl>())
+ {
+ bb = createImplicitThisMemberExpr(
+ GetTargetType(extensionDeclRef),
+ loc);
+ }
+ else
+ {
+ auto type = DeclRefType::Create(getSession(), breadcrumb->declRef);
+ bb = createImplicitThisMemberExpr(
+ type,
+ loc);
+ }
+ }
+ break;
+
default:
SLANG_UNREACHABLE("all cases handle");
}
diff --git a/source/slang/lookup.cpp b/source/slang/lookup.cpp
index cf51ae720..c0cb657c4 100644
--- a/source/slang/lookup.cpp
+++ b/source/slang/lookup.cpp
@@ -319,40 +319,35 @@ void DoLookupImpl(
if(!containerDecl)
continue;
- // If the container is a generic, then we need to instantiate it
- // at the parameters themselves, so provide a fully-resolved
- // declaration reference for lookup.
- RefPtr<Substitutions> subst = nullptr;
-#if 1
- // Actually, the above rationale seems bogus. If we are looking
- // up from "inside" a generic declaration, we don't want to
- // get its members pre-specialized, right?
-#else
- if(auto parentGenericDecl = dynamic_cast<GenericDecl*>(containerDecl->ParentDecl))
+ DeclRef<ContainerDecl> containerDeclRef =
+ DeclRef<Decl>(containerDecl, nullptr).As<ContainerDecl>();
+
+ BreadcrumbInfo breadcrumb;
+ BreadcrumbInfo* breadcrumbs = nullptr;
+
+ // Depending on the kind of container we are looking into,
+ // we may need to insert something like a `this` expression
+ // to resolve the lookup result.
+ //
+ // Note: We are checking for `AggTypeDeclBase` here, and not
+ // just `AggTypeDecl`, because we want to catch `extension`
+ // declarations as well.
+ //
+ if (auto aggTypeDeclRef = containerDeclRef.As<AggTypeDeclBase>())
{
- subst = new Substitutions();
- subst->genericDecl = parentGenericDecl;
-
- for( auto pp : parentGenericDecl->Members )
- {
- if( auto genericTypeParam = pp.As<GenericTypeParamDecl>() )
- {
- subst->args.Add(DeclRefType::Create(
- session,
- DeclRef<GenericTypeParamDecl>(genericTypeParam.Ptr(), nullptr)));
- }
- else if( auto genericValParam = pp.As<GenericValueParamDecl>() )
- {
- subst->args.Add(new GenericParamIntVal(DeclRef<GenericValueParamDecl>(genericValParam.Ptr(), nullptr)));
- }
- }
+ breadcrumb.kind = LookupResultItem::Breadcrumb::Kind::This;
+ breadcrumb.declRef = aggTypeDeclRef;
+ breadcrumb.prev = nullptr;
+
+ breadcrumbs = &breadcrumb;
}
-#endif
- DeclRef<ContainerDecl> containerRef = DeclRef<Decl>(containerDecl, subst).As<ContainerDecl>();
+ // Now perform "local" lookup in the context of the container,
+ // as if we were looking up a member directly.
+ //
DoLocalLookupImpl(
session,
- name, containerRef, request, result, nullptr);
+ name, containerDeclRef, request, result, breadcrumbs);
}
if (result.isValid())
diff --git a/source/slang/syntax.h b/source/slang/syntax.h
index d32c4de15..0b5a38631 100644
--- a/source/slang/syntax.h
+++ b/source/slang/syntax.h
@@ -859,6 +859,12 @@ namespace Slang
// lookup was able to find a member through type `U`
// instead.
Constraint,
+
+ // The lookup process considered a member of an
+ // enclosing type as being in scope, so that any
+ // reference to that member needs to use a `this`
+ // expression as appropriate.
+ This,
};
// The kind of lookup step that was performed