Skip to content

Commit 226e852

Browse files
committedJun 19, 2020
8242214: NullPointerException in JDK 14 javac compiling a method reference
Ensuring a proper receiver is used to invoke protected method when method reference is unrolled into a lambda. Reviewed-by: vromero
1 parent 5d1255b commit 226e852

File tree

3 files changed

+40
-19
lines changed

3 files changed

+40
-19
lines changed
 

‎src/jdk.compiler/share/classes/com/sun/tools/javac/comp/LambdaToMethod.java

+11-7
Original file line numberDiff line numberDiff line change
@@ -999,7 +999,10 @@ JCExpression getReceiverExpression() {
999999
private JCExpression makeReceiver(VarSymbol rcvr) {
10001000
if (rcvr == null) return null;
10011001
JCExpression rcvrExpr = make.Ident(rcvr);
1002-
Type rcvrType = tree.ownerAccessible ? tree.sym.enclClass().type : tree.expr.type;
1002+
boolean protAccess =
1003+
isProtectedInSuperClassOfEnclosingClassInOtherPackage(tree.sym, owner);
1004+
Type rcvrType = tree.ownerAccessible && !protAccess ? tree.sym.enclClass().type
1005+
: tree.expr.type;
10031006
if (rcvrType == syms.arrayClass.type) {
10041007
// Map the receiver type to the actually type, not just "array"
10051008
rcvrType = tree.getQualifierExpression().type;
@@ -2270,11 +2273,6 @@ boolean isPrivateInOtherClass() {
22702273
types.erasure(owner.enclClass().asType()));
22712274
}
22722275

2273-
boolean isProtectedInSuperClassOfEnclosingClassInOtherPackage() {
2274-
return ((tree.sym.flags() & PROTECTED) != 0 &&
2275-
tree.sym.packge() != owner.packge());
2276-
}
2277-
22782276
/**
22792277
* Erasure destroys the implementation parameter subtype
22802278
* relationship for intersection types.
@@ -2311,7 +2309,7 @@ final boolean needsConversionToLambda() {
23112309
needsVarArgsConversion() ||
23122310
isArrayOp() ||
23132311
(!nestmateLambdas && isPrivateInOtherClass()) ||
2314-
isProtectedInSuperClassOfEnclosingClassInOtherPackage() ||
2312+
isProtectedInSuperClassOfEnclosingClassInOtherPackage(tree.sym, owner) ||
23152313
!receiverAccessible() ||
23162314
(tree.getMode() == ReferenceMode.NEW &&
23172315
tree.kind != ReferenceKind.ARRAY_CTOR &&
@@ -2386,6 +2384,12 @@ private String classSig(Type type) {
23862384
}
23872385
}
23882386

2387+
private boolean isProtectedInSuperClassOfEnclosingClassInOtherPackage(Symbol targetReference,
2388+
Symbol currentClass) {
2389+
return ((targetReference.flags() & PROTECTED) != 0 &&
2390+
targetReference.packge() != currentClass.packge());
2391+
}
2392+
23892393
/**
23902394
* Signature Generation
23912395
*/

‎test/langtools/tools/javac/lambda/methodReference/ProtectedInaccessibleMethodRefTest2.java

+25-12
Original file line numberDiff line numberDiff line change
@@ -23,45 +23,58 @@
2323

2424
/*
2525
* @test
26-
* @bug 8234729
26+
* @bug 8234729 8242214
2727
* @summary Javac should eagerly change code generation for method references to avert IllegalAccessError in future.
28+
* @compile ProtectedInaccessibleMethodRefTest2.java
2829
* @run main ProtectedInaccessibleMethodRefTest2
2930
*/
3031

3132
import pack.I;
3233
import pack.J;
3334

35+
import java.lang.reflect.Method;
3436
import java.nio.file.Path;
3537
import java.nio.file.Paths;
3638
import java.util.function.Function;
37-
import java.lang.reflect.Method;
38-
import java.util.concurrent.Callable;
39+
import java.util.Arrays;
40+
import java.util.HashSet;
41+
import java.util.List;
42+
import java.util.Set;
43+
import java.util.function.BiFunction;
3944

4045
public final class ProtectedInaccessibleMethodRefTest2 extends I {
4146

4247
public static void main(String... args) {
4348
ProtectedInaccessibleMethodRefTest2 m = new ProtectedInaccessibleMethodRefTest2();
4449
m.test(Paths.get("test"));
45-
// Verify that the method reference has been folded into a lambda.
46-
boolean lambdaFound = false;
50+
// Verify that the method references have been folded into lambdas:
51+
Set<String> methodNames = new HashSet<>();
4752
for (Method meth : ProtectedInaccessibleMethodRefTest2.class.getDeclaredMethods()) {
48-
if (meth.getName().equals("lambda$test$0")) {
49-
lambdaFound = true;
50-
break;
51-
}
53+
methodNames.add(meth.getName());
5254
}
53-
if (!lambdaFound) {
55+
List<String> expectedMethods =
56+
Arrays.asList("lambda$test$0", "lambda$test$1", "lambda$test$2");
57+
if (!methodNames.containsAll(expectedMethods)) {
5458
throw new AssertionError("Did not find evidence of new code generation");
5559
}
5660
}
5761

5862
void test(Path outputDir) {
59-
Sub c = new Sub(this::readFile);
60-
c.check(outputDir);
63+
Sub c1 = new Sub(this::readFile);
64+
c1.check(outputDir);
65+
Sub c2 = new Sub(ProtectedInaccessibleMethodRefTest2::readFile, this);
66+
c2.check(outputDir);
67+
Sub c3 = new Sub(ProtectedInaccessibleMethodRefTest2::readFile2);
68+
c3.check(outputDir);
6169
}
70+
6271
public class Sub extends J {
6372
Sub(Function<Path,String> fileReader) {
6473
super(fileReader);
6574
}
75+
Sub(BiFunction<ProtectedInaccessibleMethodRefTest2, Path,String> fileReader,
76+
ProtectedInaccessibleMethodRefTest2 instance) {
77+
super(p -> fileReader.apply(instance, p));
78+
}
6679
}
6780
}

‎test/langtools/tools/javac/lambda/methodReference/pack/I.java

+4
Original file line numberDiff line numberDiff line change
@@ -29,4 +29,8 @@ public class I {
2929
protected String readFile(Path file) {
3030
return file.toString();
3131
}
32+
33+
protected static String readFile2(Path file) {
34+
return file.toString();
35+
}
3236
}

0 commit comments

Comments
 (0)
Please sign in to comment.