Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Allow revealing using a static receiver #5760

Merged
merged 7 commits into from
Sep 10, 2024
Merged
Show file tree
Hide file tree
Changes from 4 commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
Original file line number Diff line number Diff line change
Expand Up @@ -96,7 +96,7 @@ public void Resolve(PreTypeResolver resolver, ResolutionContext resolutionContex
if (effectiveExpr is NameSegment) {
resolver.ResolveNameSegment((NameSegment)effectiveExpr, true, null, resolutionContext, true);
} else {
resolver.ResolveDotSuffix((ExprDotName)effectiveExpr, true, null, resolutionContext, true);
resolver.ResolveDotSuffix((ExprDotName)effectiveExpr, true, true, null, resolutionContext, true);
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

These boolean flags are hard to review on GitHub, Non blocking for this one, but I would encourage in the future that we switch to enums when it's not obvious what a boolean is about. For example, StaticReferenceToInstance.Allowed.

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I'm a bit conflicted on this. There's also a third option of adding the argument name, so it would say allowStaticReferenceToInstance: true. Possibly you could configure a linter to always require the argument name for booleans.

}
var callee = (MemberSelectExpr)((ConcreteSyntaxExpression)effectiveExpr).ResolvedExpression;
if (callee == null) {
Expand All @@ -110,7 +110,7 @@ public void Resolve(PreTypeResolver resolver, ResolutionContext resolutionContex
if (exprClone is NameSegment) {
resolver.ResolveNameSegment((NameSegment)exprClone, true, null, revealResolutionContext, true);
} else {
resolver.ResolveDotSuffix((ExprDotName)exprClone, true, null, revealResolutionContext, true);
resolver.ResolveDotSuffix((ExprDotName)exprClone, true, true, null, revealResolutionContext, true);
}

var revealCallee = ((MemberSelectExpr)((ConcreteSyntaxExpression)exprClone).ResolvedExpression);
Expand Down
10 changes: 5 additions & 5 deletions Source/DafnyCore/Resolver/PreType/PreTypeResolve.Expressions.cs
Original file line number Diff line number Diff line change
Expand Up @@ -187,7 +187,7 @@ public void ResolveExpression(Expression expr, ResolutionContext resolutionConte
}
case ExprDotName name: {
var e = name;
ResolveDotSuffix(e, true, null, resolutionContext, false);
ResolveDotSuffix(e, false, true, null, resolutionContext, false);
if (e.PreType is PreTypePlaceholderModule) {
ReportError(e.tok, "name of module ({0}) is used as a variable", e.SuffixName);
ResetTypeAssignment(e); // the rest of type checking assumes actual types
Expand Down Expand Up @@ -1446,7 +1446,7 @@ private bool ResolveDatatypeConstructor(NameSegment expr, List<ActualBinding>/*?
/// <param name="resolutionContext"></param>
/// <param name="allowMethodCall">If false, generates an error if the name denotes a method. If true and the name denotes a method, returns
/// a Resolver_MethodCall.</param>
public Expression ResolveDotSuffix(ExprDotName expr, bool isLastNameSegment, List<ActualBinding> args, ResolutionContext resolutionContext, bool allowMethodCall) {
public Expression ResolveDotSuffix(ExprDotName expr, bool allowStaticReferenceToInstance, bool isLastNameSegment, List<ActualBinding> args, ResolutionContext resolutionContext, bool allowMethodCall) {
Contract.Requires(expr != null);
Contract.Requires(!expr.WasResolved());
Contract.Requires(resolutionContext != null);
Expand All @@ -1458,7 +1458,7 @@ public Expression ResolveDotSuffix(ExprDotName expr, bool isLastNameSegment, Lis
if (expr.Lhs is NameSegment) {
ResolveNameSegment((NameSegment)expr.Lhs, false, null, nonRevealOpts, false);
} else if (expr.Lhs is ExprDotName) {
ResolveDotSuffix((ExprDotName)expr.Lhs, false, null, nonRevealOpts, false);
ResolveDotSuffix((ExprDotName)expr.Lhs, false, false, null, nonRevealOpts, false);
} else {
ResolveExpression(expr.Lhs, nonRevealOpts);
}
Expand Down Expand Up @@ -1561,7 +1561,7 @@ public Expression ResolveDotSuffix(ExprDotName expr, bool isLastNameSegment, Lis
if (!resolver.VisibleInScope(member)) {
ReportError(expr.tok, $"member '{name}' has not been imported in this scope and cannot be accessed here");
}
if (!member.IsStatic) {
if (!member.IsStatic && !allowStaticReferenceToInstance) {
ReportError(expr.tok, $"accessing member '{name}' requires an instance expression"); //TODO Unify with similar error messages
// nevertheless, continue creating an expression that approximates a correct one
}
Expand Down Expand Up @@ -1720,7 +1720,7 @@ public MethodCallInformation ResolveApplySuffix(ApplySuffix e, ResolutionContext
r = ResolveNameSegment((NameSegment)e.Lhs, true, e.Bindings.ArgumentBindings, resolutionContext, allowMethodCall);
// note, if r is non-null, then e.Args have been resolved and r is a resolved expression that incorporates e.Args
} else if (e.Lhs is ExprDotName) {
r = ResolveDotSuffix((ExprDotName)e.Lhs, true, e.Bindings.ArgumentBindings, resolutionContext, allowMethodCall);
r = ResolveDotSuffix((ExprDotName)e.Lhs, false, true, e.Bindings.ArgumentBindings, resolutionContext, allowMethodCall);
// note, if r is non-null, then e.Args have been resolved and r is a resolved expression that incorporates e.Args
} else {
ResolveExpression(e.Lhs, resolutionContext);
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -1202,7 +1202,7 @@ public void ResolveTypeRhs(TypeRhs rr, Statement stmt, ResolutionContext resolut
PreType = rr.PreType
};
var callLhs = new ExprDotName(((UserDefinedType)rr.EType).tok, lhs, initCallName, ret?.LastComponent.OptTypeArguments);
ResolveDotSuffix(callLhs, true, rr.Bindings.ArgumentBindings, resolutionContext, true);
ResolveDotSuffix(callLhs, false, true, rr.Bindings.ArgumentBindings, resolutionContext, true);
if (prevErrorCount == ErrorCount) {
Contract.Assert(callLhs.ResolvedExpression is MemberSelectExpr); // since ResolveApplySuffix succeeded and call.Lhs denotes an expression (not a module or a type)
var methodSel = (MemberSelectExpr)callLhs.ResolvedExpression;
Expand Down
Original file line number Diff line number Diff line change
@@ -0,0 +1,39 @@
// RUN: %resolve --type-system-refresh --allow-axioms %s > %t
// RUN: %diff "%s.expect" "%t"

module Prod {
class Foo {
function P(x: int): bool {
true
}

static function Q(x: int): bool {
true
}
}

method StaticRevealWorks() {
hide *;

reveal Foo.P;
reveal Foo.Q;
}

method InstanceRevealWorks(foo: Foo) {
hide *;

reveal foo.P;
reveal foo.Q;
}
}

module Cons {
import Prod

method StaticRevealWorks() {
hide *;

reveal Prod.Foo.P;
reveal Prod.Foo.Q;
}
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,2 @@

Dafny program verifier did not attempt verification
1 change: 1 addition & 0 deletions docs/dev/news/5760.fix
Original file line number Diff line number Diff line change
@@ -0,0 +1 @@
Enable revealing a function that's a member of a type, using a static receiver
keyboardDrummer marked this conversation as resolved.
Show resolved Hide resolved
Loading