1
1
/*
2
- * Copyright (c) 2018, 2024 , Oracle and/or its affiliates. All rights reserved.
2
+ * Copyright (c) 2018, 2025 , Oracle and/or its affiliates. All rights reserved.
3
3
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
4
4
*
5
5
* This code is free software; you can redistribute it and/or modify it
23
23
24
24
/*
25
25
* @test
26
- * @bug 8205418 8207229 8207230 8230847 8245786 8247334 8248641 8240658 8246774 8274347
26
+ * @bug 8205418 8207229 8207230 8230847 8245786 8247334 8248641 8240658 8246774 8274347 8347989
27
27
* @summary Test the outcomes from Trees.getScope
28
28
* @modules jdk.compiler/com.sun.tools.javac.api
29
29
* jdk.compiler/com.sun.tools.javac.comp
52
52
import com .sun .source .tree .LambdaExpressionTree ;
53
53
import com .sun .source .tree .MethodInvocationTree ;
54
54
import com .sun .source .tree .MethodTree ;
55
+ import com .sun .source .tree .ReturnTree ;
55
56
import com .sun .source .tree .Scope ;
56
57
import com .sun .source .tree .Tree ;
57
58
import com .sun .source .tree .VariableTree ;
73
74
import com .sun .tools .javac .tree .JCTree .JCStatement ;
74
75
import com .sun .tools .javac .util .Context ;
75
76
import com .sun .tools .javac .util .Context .Factory ;
77
+ import java .util .Objects ;
76
78
import javax .lang .model .element .TypeElement ;
79
+ import javax .lang .model .type .DeclaredType ;
80
+ import javax .lang .model .type .TypeKind ;
81
+ import javax .lang .model .type .TypeMirror ;
82
+ import javax .lang .model .type .TypeVariable ;
77
83
import javax .lang .model .util .ElementFilter ;
78
84
import javax .tools .JavaFileObject ;
79
85
@@ -93,6 +99,7 @@ public static void main(String... args) throws IOException {
93
99
new TestGetScopeResult ().testRuleCases ();
94
100
new TestGetScopeResult ().testNestedSwitchExpression ();
95
101
new TestGetScopeResult ().testModuleImportScope ();
102
+ new TestGetScopeResult ().testClassTypeSetInEnterGetScope ();
96
103
}
97
104
98
105
public void run () throws IOException {
@@ -885,6 +892,80 @@ class Test {
885
892
}
886
893
}
887
894
895
+ //JDK-8347989
896
+ void testClassTypeSetInEnterGetScope () throws IOException {
897
+ JavacTool c = JavacTool .create ();
898
+ try (StandardJavaFileManager fm = c .getStandardFileManager (null , null , null )) {
899
+ String code = """
900
+ import java.util.*;
901
+ class Test<T extends Test&CharSequence> extends ArrayList
902
+ implements List {
903
+ private int test(boolean b) {
904
+ int v = b ? test(!b) : 0;
905
+ return v;
906
+ }
907
+ }
908
+ """ ;
909
+ Context ctx = new Context ();
910
+ TestAnalyzer .preRegister (ctx );
911
+ JavaFileObject input =
912
+ SimpleJavaFileObject .forSource (URI .create ("myfo:///Test.java" ), code );
913
+ JavacTask t = (JavacTask ) c .getTask (null , fm , null , null , null ,
914
+ List .of (input ),
915
+ ctx );
916
+ Trees trees = Trees .instance (t );
917
+ List <List <String >> actual = new ArrayList <>();
918
+
919
+ t .addTaskListener (new TaskListener () {
920
+ @ Override
921
+ public void finished (TaskEvent e ) {
922
+ if (e .getKind () != TaskEvent .Kind .ENTER ) {
923
+ return ;
924
+ }
925
+
926
+ new TreePathScanner <Void , Void >() {
927
+ @ Override
928
+ public Void visitClass (ClassTree node , Void p ) {
929
+ TypeMirror type = trees .getTypeMirror (getCurrentPath ());
930
+ if (type == null ) {
931
+ throw new AssertionError ("Expected class type 'Test', but got: null" );
932
+ }
933
+ assertEquals (TypeKind .DECLARED , type .getKind ());
934
+ DeclaredType decl = (DeclaredType ) type ;
935
+ TypeVariable tvar = (TypeVariable ) decl .getTypeArguments ().get (0 );
936
+ assertEquals ("T" , tvar .asElement ().getSimpleName ().toString ());
937
+ assertEquals ("Test&java.lang.CharSequence" , tvar .getUpperBound ().toString ());
938
+ TypeElement clazz = (TypeElement ) decl .asElement ();
939
+ assertEquals ("java.util.ArrayList" , clazz .getSuperclass ().toString ().toString ());
940
+ assertEquals ("java.util.List" , clazz .getInterfaces ().toString ().toString ());
941
+ return super .visitClass (node , p );
942
+ }
943
+ @ Override
944
+ public Void visitReturn (ReturnTree rt , Void p ) {
945
+ Scope scope = trees .getScope (getCurrentPath ());
946
+ actual .add (dumpScope (scope ));
947
+ return super .visitReturn (rt , p );
948
+ }
949
+ }.scan (e .getCompilationUnit (), null );
950
+ }
951
+ });
952
+
953
+ t .analyze ();
954
+
955
+ List <List <String >> expected =
956
+ List .of (List .of ("v:int" ,
957
+ "b:boolean" ,
958
+ "super:java.util.ArrayList" ,
959
+ "this:Test<T>" ,
960
+ "T:T"
961
+ ));
962
+
963
+ if (!expected .equals (actual )) {
964
+ throw new AssertionError ("Unexpected Scope content: " + actual );
965
+ }
966
+ }
967
+ }
968
+
888
969
private List <String > dumpScope (Scope scope ) {
889
970
List <String > content = new ArrayList <>();
890
971
while (scope .getEnclosingClass () != null ) {
@@ -908,4 +989,10 @@ private void asssertScopeContainsTypeWithFQN(Scope scope, String fqn) {
908
989
", but it is missing." );
909
990
}
910
991
992
+ private void assertEquals (Object expected , Object actual ) {
993
+ if (!Objects .equals (expected , actual )) {
994
+ throw new AssertionError ("Expected: '" + expected + "', " +
995
+ "but got: '" + actual + "'" );
996
+ }
997
+ }
911
998
}
0 commit comments