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
|
package cuchaz.enigma.gui;
import cuchaz.enigma.Deobfuscator;
import cuchaz.enigma.analysis.EntryReference;
import cuchaz.enigma.analysis.SourceIndex;
import cuchaz.enigma.analysis.Token;
import cuchaz.enigma.api.EnigmaPlugin;
import cuchaz.enigma.gui.highlight.TokenHighlightType;
import cuchaz.enigma.translation.LocalNameGenerator;
import cuchaz.enigma.translation.Translator;
import cuchaz.enigma.translation.representation.TypeDescriptor;
import cuchaz.enigma.translation.representation.entry.ClassEntry;
import cuchaz.enigma.translation.representation.entry.Entry;
import cuchaz.enigma.translation.representation.entry.FieldEntry;
import cuchaz.enigma.translation.representation.entry.LocalVariableDefEntry;
import javax.annotation.Nullable;
import java.util.*;
public class DecompiledClassSource {
private final ClassEntry classEntry;
private final Deobfuscator deobfuscator;
private final SourceIndex obfuscatedIndex;
private SourceIndex remappedIndex;
private final Map<TokenHighlightType, Collection<Token>> highlightedTokens = new EnumMap<>(TokenHighlightType.class);
public DecompiledClassSource(ClassEntry classEntry, Deobfuscator deobfuscator, SourceIndex index) {
this.classEntry = classEntry;
this.deobfuscator = deobfuscator;
this.obfuscatedIndex = index;
this.remappedIndex = index;
}
public void remapSource(Translator translator) {
highlightedTokens.clear();
SourceRemapper remapper = new SourceRemapper(obfuscatedIndex.getSource(), obfuscatedIndex.referenceTokens());
SourceRemapper.Result remapResult = remapper.remap((token, movedToken) -> remapToken(token, movedToken, translator));
remappedIndex = obfuscatedIndex.remapTo(remapResult);
}
private String remapToken(Token token, Token movedToken, Translator translator) {
EntryReference<Entry<?>, Entry<?>> reference = obfuscatedIndex.getReference(token);
if (deobfuscator.isRenamable(reference)) {
Entry<?> entry = reference.getNameableEntry();
Entry<?> translatedEntry = translator.translate(entry);
if (isDeobfuscated(entry, translatedEntry)) {
highlightToken(movedToken, TokenHighlightType.DEOBFUSCATED);
return translatedEntry.getSourceRemapName();
} else {
String proposedName = proposeName(entry);
if (proposedName != null) {
highlightToken(movedToken, TokenHighlightType.PROPOSED);
return proposedName;
}
highlightToken(movedToken, TokenHighlightType.OBFUSCATED);
String defaultName = generateDefaultName(translatedEntry);
if (defaultName != null) {
return defaultName;
}
}
}
return null;
}
@Nullable
private String proposeName(Entry<?> entry) {
if (entry instanceof FieldEntry) {
for (EnigmaPlugin plugin : deobfuscator.getPlugins()) {
String owner = entry.getContainingClass().getFullName();
String proposal = plugin.proposeFieldName(owner, entry.getName(), ((FieldEntry) entry).getDesc().toString());
if (proposal != null) {
return proposal;
}
}
}
return null;
}
@Nullable
private String generateDefaultName(Entry<?> entry) {
if (entry instanceof LocalVariableDefEntry) {
LocalVariableDefEntry localVariable = (LocalVariableDefEntry) entry;
int index = localVariable.getIndex();
if (localVariable.isArgument()) {
List<TypeDescriptor> arguments = localVariable.getParent().getDesc().getArgumentDescs();
return LocalNameGenerator.generateArgumentName(index, localVariable.getDesc(), arguments);
} else {
return LocalNameGenerator.generateLocalVariableName(index, localVariable.getDesc());
}
}
return null;
}
private boolean isDeobfuscated(Entry<?> entry, Entry<?> translatedEntry) {
return !entry.getName().equals(translatedEntry.getName());
}
public ClassEntry getEntry() {
return classEntry;
}
public SourceIndex getIndex() {
return remappedIndex;
}
public Map<TokenHighlightType, Collection<Token>> getHighlightedTokens() {
return highlightedTokens;
}
private void highlightToken(Token token, TokenHighlightType highlightType) {
highlightedTokens.computeIfAbsent(highlightType, t -> new ArrayList<>()).add(token);
}
@Override
public String toString() {
return remappedIndex.getSource();
}
}
|