-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathbytecode-utils.kt
150 lines (127 loc) · 4.75 KB
/
bytecode-utils.kt
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
package me.hydos.fukkitdevelopment
import com.intellij.openapi.project.Project
import com.intellij.psi.JavaPsiFacade
import com.intellij.psi.PsiArrayType
import com.intellij.psi.PsiClass
import com.intellij.psi.PsiClassType
import com.intellij.psi.PsiField
import com.intellij.psi.PsiMethod
import com.intellij.psi.PsiPrimitiveType
import com.intellij.psi.PsiType
import com.intellij.psi.search.GlobalSearchScope
import com.intellij.psi.util.TypeConversionUtil
private const val INTERNAL_CONSTRUCTOR_NAME = "<init>"
// Type
val PsiPrimitiveType.internalName: Char
get() = when (this) {
PsiType.BYTE -> 'B'
PsiType.CHAR -> 'C'
PsiType.DOUBLE -> 'D'
PsiType.FLOAT -> 'F'
PsiType.INT -> 'I'
PsiType.LONG -> 'J'
PsiType.SHORT -> 'S'
PsiType.BOOLEAN -> 'Z'
PsiType.VOID -> 'V'
else -> throw IllegalArgumentException("Unsupported primitive type: $this")
}
fun getPrimitiveType(internalName: Char): PsiPrimitiveType? {
return when (internalName) {
'B' -> PsiType.BYTE
'C' -> PsiType.CHAR
'D' -> PsiType.DOUBLE
'F' -> PsiType.FLOAT
'I' -> PsiType.INT
'J' -> PsiType.LONG
'S' -> PsiType.SHORT
'Z' -> PsiType.BOOLEAN
'V' -> PsiType.VOID
else -> null
}
}
fun getPrimitiveWrapperClass(internalName: Char, project: Project): PsiClass? {
val type = getPrimitiveType(internalName) ?: return null
val boxedTypeName = type.boxedTypeName ?: return null
return JavaPsiFacade.getInstance(project).findClass(boxedTypeName, GlobalSearchScope.allScope(project))
}
private fun PsiClassType.erasure() = TypeConversionUtil.erasure(this) as PsiClassType
@Throws(ClassNameResolutionFailedException::class)
private fun PsiClassType.appendInternalName(builder: StringBuilder): StringBuilder =
erasure().resolve()?.appendInternalName(builder) ?: builder
@Throws(ClassNameResolutionFailedException::class)
private fun PsiType.appendDescriptor(builder: StringBuilder): StringBuilder {
return when (this) {
is PsiPrimitiveType -> builder.append(internalName)
is PsiArrayType -> componentType.appendDescriptor(builder.append('['))
is PsiClassType -> appendInternalName(builder.append('L')).append(';')
else -> throw IllegalArgumentException("Unsupported PsiType: $this")
}
}
fun parseClassDescriptor(descriptor: String): String {
val internalName = descriptor.substring(1, descriptor.length - 1)
return internalName.replace('/', '.')
}
// Class
val PsiClass.internalName: String?
get() {
return try {
outerQualifiedName?.replace('.', '/') ?: buildInternalName(StringBuilder()).toString()
} catch (e: ClassNameResolutionFailedException) {
null
}
}
@Throws(ClassNameResolutionFailedException::class)
private fun PsiClass.appendInternalName(builder: StringBuilder): StringBuilder {
return outerQualifiedName?.let { builder.append(it.replace('.', '/')) } ?: buildInternalName(builder)
}
@Throws(ClassNameResolutionFailedException::class)
private fun PsiClass.buildInternalName(builder: StringBuilder): StringBuilder {
buildInnerName(builder, { it.outerQualifiedName?.replace('.', '/') })
return builder
}
val PsiClass.descriptor: String?
get() {
return try {
appendInternalName(StringBuilder().append('L')).append(';').toString()
} catch (e: ClassNameResolutionFailedException) {
null
}
}
fun PsiClass.findMethodsByInternalName(internalName: String, checkBases: Boolean = false): Array<PsiMethod> {
return if (internalName == INTERNAL_CONSTRUCTOR_NAME) {
constructors
} else {
findMethodsByName(internalName, checkBases)
}
}
// Method
val PsiMethod.internalName: String
get() = if (isConstructor) INTERNAL_CONSTRUCTOR_NAME else name
val PsiMethod.descriptor: String?
get() {
return try {
appendDescriptor(StringBuilder()).toString()
} catch (e: ClassNameResolutionFailedException) {
null
}
}
@Throws(ClassNameResolutionFailedException::class)
private fun PsiMethod.appendDescriptor(builder: StringBuilder): StringBuilder {
builder.append('(')
for (parameter in parameterList.parameters) {
parameter.type.appendDescriptor(builder)
}
builder.append(')')
return (returnType ?: PsiType.VOID).appendDescriptor(builder)
}
// Field
val PsiField.descriptor: String?
get() {
return try {
appendDescriptor(StringBuilder()).toString()
} catch (e: ClassNameResolutionFailedException) {
null
}
}
@Throws(ClassNameResolutionFailedException::class)
private fun PsiField.appendDescriptor(builder: StringBuilder): StringBuilder = type.appendDescriptor(builder)