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

Warn when using the top type in a pre- or post-condition annotation #6412

Open
wants to merge 5 commits into
base: master
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from 2 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 @@ -59,6 +59,7 @@
import java.util.Map;
import java.util.Set;
import java.util.StringJoiner;
import java.util.TreeSet;
import java.util.Vector;
import javax.annotation.processing.ProcessingEnvironment;
import javax.lang.model.element.AnnotationMirror;
Expand Down Expand Up @@ -101,6 +102,11 @@
import org.checkerframework.framework.flow.CFAbstractStore;
import org.checkerframework.framework.flow.CFAbstractValue;
import org.checkerframework.framework.qual.DefaultQualifier;
import org.checkerframework.framework.qual.EnsuresQualifier;
import org.checkerframework.framework.qual.EnsuresQualifierIf;
import org.checkerframework.framework.qual.PostconditionAnnotation;
import org.checkerframework.framework.qual.PreconditionAnnotation;
import org.checkerframework.framework.qual.RequiresQualifier;
import org.checkerframework.framework.qual.Unused;
import org.checkerframework.framework.source.DiagMessage;
import org.checkerframework.framework.source.SourceVisitor;
Expand Down Expand Up @@ -2234,16 +2240,33 @@ public Void visitAnnotation(AnnotationTree tree, Void p) {
return null;
}

TypeElement anno = (TypeElement) TreeInfo.symbol((JCTree) tree.getAnnotationType());
TypeElement annoType = (TypeElement) TreeInfo.symbol((JCTree) tree.getAnnotationType());

Name annoName = anno.getQualifiedName();
Name annoName = annoType.getQualifiedName();
if (annoName.contentEquals(DefaultQualifier.class.getName())
|| annoName.contentEquals(SuppressWarnings.class.getName())) {
// Skip these two annotations, as we don't care about the arguments to them.
return null;
}

List<ExecutableElement> methods = ElementFilter.methodsIn(anno.getEnclosedElements());
if (isPreOrPostConditionAnnotation(annoName)) {
Copy link
Member

Choose a reason for hiding this comment

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

Why isn't this logic in BaseTypeVisitor#checkContractsAtMethodDeclaration?

Copy link
Contributor

Choose a reason for hiding this comment

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

I initially implemented this check in BaseTypeVisitor#checkContractsAtMethodDeclaration (because I also thought it was useful that the contracts are already there in the method), but that method appears to check implicit annotations, as well.

AnnotationMirror anno = TreeUtils.annotationFromAnnotationTree(tree);
System.out.printf("anno = %s for annoName = %s for tree = %s%n", anno, annoName, tree);
Copy link
Member

Choose a reason for hiding this comment

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

This needs to be removed or commented.

// TODO: `getQualifierEnforcedByContractAnnotation` does not work for @PreconditionAnnotation
Copy link
Member Author

Choose a reason for hiding this comment

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

@smillst What is your view on this design question? I lean toward the latter.

Copy link
Member

Choose a reason for hiding this comment

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

It's hard to answer this question because I don't know what you mean by "does not work". How/where are those annotations handled else where?

// or @PostconditionAnnotation. Should I extend the method to handle that, or should I put
// the logic here and avoid calling the method?
AnnotationMirror qualifier =
atypeFactory.getContractsFromMethod().getQualifierEnforcedByContractAnnotation(anno);
if (qualifier != null) {
AnnotationMirror topForQualifier = qualHierarchy.getTopAnnotation(qualifier);
if (AnnotationUtils.areSame(qualifier, topForQualifier)) {
Name contractQualName = qualifier.getAnnotationType().asElement().getSimpleName();
checker.reportWarning(tree, "contracts.toptype", contractQualName);
}
}
}

List<ExecutableElement> methods = ElementFilter.methodsIn(annoType.getEnclosedElements());
// Mapping from argument simple name to its annotated type.
Map<String, AnnotatedTypeMirror> annoTypes = ArrayMap.newArrayMapOrHashMap(methods.size());
for (ExecutableElement meth : methods) {
Expand Down Expand Up @@ -2299,6 +2322,29 @@ public Void visitAnnotation(AnnotationTree tree, Void p) {
return null;
}

/** Pre- and post-condition annotations that take a qualifier as an argument. */
private TreeSet<String> preAndPostConditionAnnotations =
new TreeSet<>(
// In Java 9+, use `List.of()`.
Arrays.asList(
PreconditionAnnotation.class.getName(),
PostconditionAnnotation.class.getName(),
RequiresQualifier.class.getName(),
EnsuresQualifier.class.getName(),
EnsuresQualifierIf.class.getName()));

/**
* Returns true if the given annotation name matches that of a pre- or post-condition annotation
* or meta-annotation that takes a qualifier as an argument.
*
* @param annotationName an annotation name
* @return true iff the annotation name matches that of a pre- or post-condition annotation
*/
private boolean isPreOrPostConditionAnnotation(Name annotationName) {
String annoName = annotationName.toString();
return preAndPostConditionAnnotations.contains(annoName);
}

/**
* If the computation of the type of the ConditionalExpressionTree in
* org.checkerframework.framework.type.TypeFromTree.TypeFromExpression.visitConditionalExpression(ConditionalExpressionTree,
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -88,6 +88,7 @@ flowexpr.parse.context.not.determined=could not determine the context at '%s' wi
flowexpr.parameter.not.final=parameter %s in '%s' is not effectively final (i.e., it gets re-assigned)
contracts.precondition=precondition of %s is not satisfied.%nfound : %s%nrequired: %s
contracts.postcondition=postcondition of %s is not satisfied.%nfound : %s%nrequired: %s
contracts.toptype=the top type %s has no effect
contracts.conditional.postcondition=conditional postcondition is not satisfied when %s returns %s.%nfound : %s%nrequired: %s
contracts.conditional.postcondition.returntype=this annotation is only valid for methods with return type 'boolean'
# Same text for "override" and "methodref", but different key.
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -207,10 +207,11 @@ private <T extends Contract> Set<T> getContract(
* Returns the annotation mirror as specified by the {@code qualifier} element in {@code
* contractAnno}. May return null.
*
* @param contractAnno a pre- or post-condition annotation, such as {@code @RequiresQualifier}
* @param contractAnno a pre- or post-condition annotation, such as {@code @RequiresQualifier}.
* Does not work for {@code PreconditionAnnotation} or {@code PostconditionAnnotation}.
* @return the type annotation specified in {@code contractAnno.qualifier}
*/
private @Nullable AnnotationMirror getQualifierEnforcedByContractAnnotation(
public @Nullable AnnotationMirror getQualifierEnforcedByContractAnnotation(
AnnotationMirror contractAnno) {
return getQualifierEnforcedByContractAnnotation(contractAnno, null, null);
}
Expand Down
6 changes: 6 additions & 0 deletions framework/tests/flow/ContractsOverridingSubtyping.java
Original file line number Diff line number Diff line change
Expand Up @@ -12,6 +12,7 @@ static class Base {
@RequiresQualifier(expression = "f", qualifier = Odd.class)
void requiresOdd() {}

// :: warning: (contracts.toptype)
@RequiresQualifier(expression = "f", qualifier = Unqualified.class)
void requiresUnqual() {}

Expand All @@ -20,6 +21,7 @@ void ensuresOdd() {
f = g;
}

// :: warning: (contracts.toptype)
@EnsuresQualifier(expression = "f", qualifier = Unqualified.class)
void ensuresUnqual() {}

Expand All @@ -29,6 +31,7 @@ boolean ensuresIfOdd() {
return true;
}

// :: warning: (contracts.toptype)
@EnsuresQualifierIf(expression = "f", result = true, qualifier = Unqualified.class)
boolean ensuresIfUnqual() {
return true;
Expand All @@ -38,6 +41,7 @@ boolean ensuresIfUnqual() {
static class Derived extends Base {

@Override
// :: warning: (contracts.toptype)
@RequiresQualifier(expression = "f", qualifier = Unqualified.class)
void requiresOdd() {}

Expand All @@ -47,6 +51,7 @@ void requiresOdd() {}
void requiresUnqual() {}

@Override
// :: warning: (contracts.toptype)
@EnsuresQualifier(expression = "f", qualifier = Unqualified.class)
// :: error: (contracts.postcondition.override)
void ensuresOdd() {
Expand All @@ -60,6 +65,7 @@ void ensuresUnqual() {
}

@Override
// :: warning: (contracts.toptype)
@EnsuresQualifierIf(expression = "f", result = true, qualifier = Unqualified.class)
// :: error: (contracts.conditional.postcondition.true.override)
boolean ensuresIfOdd() {
Expand Down
12 changes: 12 additions & 0 deletions framework/tests/flow/Postcondition.java
Original file line number Diff line number Diff line change
@@ -1,3 +1,4 @@
import org.checkerframework.common.subtyping.qual.Unqualified;
import org.checkerframework.dataflow.qual.Pure;
import org.checkerframework.framework.qual.EnsuresQualifier;
import org.checkerframework.framework.qual.EnsuresQualifierIf;
Expand Down Expand Up @@ -312,4 +313,15 @@ void t6(@Odd String p1, @ValueTypeAnno String p2) {
@Odd String l6 = f1;
}
}

/** *** errors for invalid postconditions ***** */
// :: warning: (contracts.toptype)
@EnsuresQualifier(expression = "f1", qualifier = Unqualified.class)
void noOpForTesting() {}

// :: warning: (contracts.toptype)
@EnsuresQualifierIf(result = true, expression = "f1", qualifier = Unqualified.class)
boolean isF1NotSet() {
return f1 == null;
}
}
5 changes: 5 additions & 0 deletions framework/tests/flow/Precondition.java
Original file line number Diff line number Diff line change
@@ -1,3 +1,4 @@
import org.checkerframework.common.subtyping.qual.Unqualified;
import org.checkerframework.dataflow.qual.Pure;
import org.checkerframework.framework.qual.RequiresQualifier;
import org.checkerframework.framework.test.*;
Expand Down Expand Up @@ -159,4 +160,8 @@ void t5(@Odd String p1, String p2, @ValueTypeAnno String p3) {
// :: error: (flowexpr.parse.error)
error2();
}

// :: warning: (contracts.toptype)
@RequiresQualifier(expression = "f1", qualifier = Unqualified.class)
void noOpForTesting() {}
}
Loading