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
|
/*******************************************************************************
* Copyright (c) 2014 Jeff Martin.
* All rights reserved. This program and the accompanying materials
* are made available under the terms of the GNU Public License v3.0
* which accompanies this distribution, and is available at
* http://www.gnu.org/licenses/gpl.html
*
* Contributors:
* Jeff Martin - initial API and implementation
******************************************************************************/
package cuchaz.enigma.analysis;
import com.strobel.assembler.metadata.FieldDefinition;
import com.strobel.assembler.metadata.MethodDefinition;
import com.strobel.assembler.metadata.TypeDefinition;
import com.strobel.assembler.metadata.TypeReference;
import com.strobel.decompiler.languages.TextLocation;
import com.strobel.decompiler.languages.java.ast.ConstructorDeclaration;
import com.strobel.decompiler.languages.java.ast.EnumValueDeclaration;
import com.strobel.decompiler.languages.java.ast.FieldDeclaration;
import com.strobel.decompiler.languages.java.ast.Keys;
import com.strobel.decompiler.languages.java.ast.MethodDeclaration;
import com.strobel.decompiler.languages.java.ast.SimpleType;
import com.strobel.decompiler.languages.java.ast.TypeDeclaration;
import com.strobel.decompiler.languages.java.ast.VariableInitializer;
import cuchaz.enigma.mapping.BehaviorEntry;
import cuchaz.enigma.mapping.ClassEntry;
import cuchaz.enigma.mapping.ConstructorEntry;
import cuchaz.enigma.mapping.Entry;
import cuchaz.enigma.mapping.FieldEntry;
import cuchaz.enigma.mapping.MethodEntry;
public class SourceIndexClassVisitor extends SourceIndexVisitor
{
private ClassEntry m_classEntry;
public SourceIndexClassVisitor( ClassEntry classEntry )
{
m_classEntry = classEntry;
}
@Override
public Void visitTypeDeclaration( TypeDeclaration node, SourceIndex index )
{
// is this this class, or a subtype?
TypeDefinition def = node.getUserData( Keys.TYPE_DEFINITION );
ClassEntry classEntry = new ClassEntry( def.getInternalName() );
if( !classEntry.equals( m_classEntry ) )
{
// it's a sub-type, recurse
index.addDeclaration( node.getNameToken(), classEntry );
return node.acceptVisitor( new SourceIndexClassVisitor( classEntry ), index );
}
return recurse( node, index );
}
@Override
public Void visitSimpleType( SimpleType node, SourceIndex index )
{
TypeReference ref = node.getUserData( Keys.TYPE_REFERENCE );
if( node.getIdentifierToken().getStartLocation() != TextLocation.EMPTY )
{
ClassEntry classEntry = new ClassEntry( ref.getInternalName() );
index.addReference(
node.getIdentifierToken(),
new EntryReference<Entry,Entry>( classEntry, m_classEntry )
);
}
return recurse( node, index );
}
@Override
public Void visitMethodDeclaration( MethodDeclaration node, SourceIndex index )
{
MethodDefinition def = node.getUserData( Keys.METHOD_DEFINITION );
ClassEntry classEntry = new ClassEntry( def.getDeclaringType().getInternalName() );
BehaviorEntry behaviorEntry;
if( def.getName().equals( "<clinit>" ) )
{
behaviorEntry = new ConstructorEntry( classEntry );
}
else
{
behaviorEntry = new MethodEntry( classEntry, def.getName(), def.getSignature() );
}
index.addDeclaration( node.getNameToken(), behaviorEntry );
return node.acceptVisitor( new SourceIndexBehaviorVisitor( behaviorEntry ), index );
}
@Override
public Void visitConstructorDeclaration( ConstructorDeclaration node, SourceIndex index )
{
MethodDefinition def = node.getUserData( Keys.METHOD_DEFINITION );
ClassEntry classEntry = new ClassEntry( def.getDeclaringType().getInternalName() );
ConstructorEntry constructorEntry = new ConstructorEntry( classEntry, def.getSignature() );
index.addDeclaration( node.getNameToken(), constructorEntry );
return node.acceptVisitor( new SourceIndexBehaviorVisitor( constructorEntry ), index );
}
@Override
public Void visitFieldDeclaration( FieldDeclaration node, SourceIndex index )
{
FieldDefinition def = node.getUserData( Keys.FIELD_DEFINITION );
ClassEntry classEntry = new ClassEntry( def.getDeclaringType().getInternalName() );
FieldEntry fieldEntry = new FieldEntry( classEntry, def.getName() );
assert( node.getVariables().size() == 1 );
VariableInitializer variable = node.getVariables().firstOrNullObject();
index.addDeclaration( variable.getNameToken(), fieldEntry );
return recurse( node, index );
}
@Override
public Void visitEnumValueDeclaration( EnumValueDeclaration node, SourceIndex index )
{
// treat enum declarations as field declarations
FieldDefinition def = node.getUserData( Keys.FIELD_DEFINITION );
ClassEntry classEntry = new ClassEntry( def.getDeclaringType().getInternalName() );
FieldEntry fieldEntry = new FieldEntry( classEntry, def.getName() );
index.addDeclaration( node.getNameToken(), fieldEntry );
return recurse( node, index );
}
}
|