1
1
/*
2
- * Copyright (c) 2012, 2019 , Oracle and/or its affiliates. All rights reserved.
2
+ * Copyright (c) 2012, 2022 , 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
27
27
*/
28
28
public class MetaUtil {
29
29
30
+ public static final char PACKAGE_SEPARATOR_INTERNAL = '/' ;
31
+ public static final char HIDDEN_SEPARATOR_INTERNAL = '.' ;
32
+ public static final char PACKAGE_SEPARATOR_JAVA = HIDDEN_SEPARATOR_INTERNAL ;
33
+ public static final char HIDDEN_SEPARATOR_JAVA = PACKAGE_SEPARATOR_INTERNAL ;
34
+
30
35
/**
31
36
* Extends the functionality of {@link Class#getSimpleName()} to include a non-empty string for
32
37
* anonymous and local classes.
@@ -87,25 +92,27 @@ private static String safeSimpleName(Class<?> clazz) {
87
92
}
88
93
89
94
/**
90
- * Classes for lambdas can have {@code /} characters that are not package separators. These are
91
- * distinguished by being followed by a character that is not a
95
+ * Hidden classes have {@code /} characters in their internal names and {@code .} characters in their names returned
96
+ * by {@link Class#getName()} that are not package separators.
97
+ * These are distinguished by being followed by a character that is not a
92
98
* {@link Character#isJavaIdentifierStart(char)} (e.g.,
93
99
* "jdk.vm.ci.runtime.test.TypeUniverse$$Lambda$1/869601985").
100
+ *
101
+ * @param name the name to perform the replacements on
102
+ * @param packageSeparator the {@link Character} used as the package separator, e.g. {@code /} in internal form
103
+ * @param hiddenSeparator the {@link Character} used as the hidden class separator, e.g. {@code .} in internal form
94
104
*/
95
- private static String replacePackageSeparatorsWithDot (String name ) {
105
+ private static String replacePackageAndHiddenSeparators (String name , Character packageSeparator , Character hiddenSeparator ) {
106
+ int index = name .indexOf (hiddenSeparator ); // check if it's a hidden class
96
107
int length = name .length ();
97
- int i = 0 ;
98
108
StringBuilder buf = new StringBuilder (length );
99
- while (i < length - 1 ) {
100
- char ch = name .charAt (i );
101
- if (ch == '/' && Character .isJavaIdentifierStart (name .charAt (i + 1 ))) {
102
- buf .append ('.' );
103
- } else {
104
- buf .append (ch );
105
- }
106
- i ++;
109
+ if (index < 0 ) {
110
+ buf .append (name .replace (packageSeparator , hiddenSeparator ));
111
+ } else {
112
+ buf .append (name .substring (0 , index ).replace (packageSeparator , hiddenSeparator ));
113
+ buf .append (packageSeparator );
114
+ buf .append (name .substring (index + 1 ));
107
115
}
108
- buf .append (name .charAt (length - 1 ));
109
116
return buf .toString ();
110
117
}
111
118
@@ -122,17 +129,22 @@ private static String replacePackageSeparatorsWithDot(String name) {
122
129
public static String internalNameToJava (String name , boolean qualified , boolean classForNameCompatible ) {
123
130
switch (name .charAt (0 )) {
124
131
case 'L' : {
125
- String result = replacePackageSeparatorsWithDot (name .substring (1 , name .length () - 1 ));
132
+ String type = name .substring (1 , name .length () - 1 );
133
+ String result = replacePackageAndHiddenSeparators (type , PACKAGE_SEPARATOR_INTERNAL , HIDDEN_SEPARATOR_INTERNAL );
126
134
if (!qualified ) {
127
- final int lastDot = result .lastIndexOf ('.' );
135
+ final int lastDot = result .lastIndexOf (HIDDEN_SEPARATOR_INTERNAL );
128
136
if (lastDot != -1 ) {
129
137
result = result .substring (lastDot + 1 );
130
138
}
131
139
}
132
140
return result ;
133
141
}
134
142
case '[' :
135
- return classForNameCompatible ? replacePackageSeparatorsWithDot (name ) : internalNameToJava (name .substring (1 ), qualified , classForNameCompatible ) + "[]" ;
143
+ if (classForNameCompatible ) {
144
+ return replacePackageAndHiddenSeparators (name , PACKAGE_SEPARATOR_INTERNAL , HIDDEN_SEPARATOR_INTERNAL );
145
+ } else {
146
+ return internalNameToJava (name .substring (1 ), qualified , false ) + "[]" ;
147
+ }
136
148
default :
137
149
if (name .length () != 1 ) {
138
150
throw new IllegalArgumentException ("Illegal internal name: " + name );
@@ -213,7 +225,7 @@ static void appendProfile(StringBuilder buf, AbstractJavaProfile<?, ?> profile,
213
225
public static String toInternalName (String className ) {
214
226
if (className .startsWith ("[" )) {
215
227
/* Already in the correct array style. */
216
- return className . replace ( '.' , '/' );
228
+ return replacePackageAndHiddenSeparators ( className , PACKAGE_SEPARATOR_JAVA , HIDDEN_SEPARATOR_JAVA );
217
229
}
218
230
219
231
StringBuilder result = new StringBuilder ();
@@ -252,7 +264,9 @@ public static String toInternalName(String className) {
252
264
result .append ("V" );
253
265
break ;
254
266
default :
255
- result .append ("L" ).append (base .replace ('.' , '/' )).append (";" );
267
+ result .append ("L" )
268
+ .append (replacePackageAndHiddenSeparators (base , PACKAGE_SEPARATOR_JAVA , HIDDEN_SEPARATOR_JAVA ))
269
+ .append (";" );
256
270
break ;
257
271
}
258
272
return result .toString ();
0 commit comments