mirror of
https://github.com/Eaglercraft-TeaVM-Fork/eagler-teavm.git
synced 2024-12-22 16:14:10 -08:00
C: add stack trace deobfuscation tool
This commit is contained in:
parent
e164015bfa
commit
099b9c5827
258
core/src/main/java/org/teavm/backend/c/util/Deobfuscator.java
Normal file
258
core/src/main/java/org/teavm/backend/c/util/Deobfuscator.java
Normal file
|
@ -0,0 +1,258 @@
|
|||
/*
|
||||
* Copyright 2019 konsoletyper.
|
||||
*
|
||||
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||
* you may not use this file except in compliance with the License.
|
||||
* You may obtain a copy of the License at
|
||||
*
|
||||
* http://www.apache.org/licenses/LICENSE-2.0
|
||||
*
|
||||
* Unless required by applicable law or agreed to in writing, software
|
||||
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||
* See the License for the specific language governing permissions and
|
||||
* limitations under the License.
|
||||
*/
|
||||
package org.teavm.backend.c.util;
|
||||
|
||||
import java.io.BufferedInputStream;
|
||||
import java.io.FileInputStream;
|
||||
import java.io.IOException;
|
||||
import java.io.InputStream;
|
||||
import java.io.InputStreamReader;
|
||||
import java.io.Reader;
|
||||
import java.nio.charset.StandardCharsets;
|
||||
import java.util.ArrayList;
|
||||
import java.util.HashMap;
|
||||
import java.util.List;
|
||||
import java.util.Map;
|
||||
import org.teavm.backend.c.util.json.JsonAllErrorVisitor;
|
||||
import org.teavm.backend.c.util.json.JsonArrayVisitor;
|
||||
import org.teavm.backend.c.util.json.JsonErrorReporter;
|
||||
import org.teavm.backend.c.util.json.JsonParser;
|
||||
import org.teavm.backend.c.util.json.JsonPropertyVisitor;
|
||||
import org.teavm.backend.c.util.json.JsonVisitingConsumer;
|
||||
import org.teavm.backend.c.util.json.JsonVisitor;
|
||||
|
||||
public class Deobfuscator {
|
||||
private Map<Integer, CallSite> callSites = new HashMap<>();
|
||||
|
||||
public Deobfuscator(Reader reader) throws IOException {
|
||||
CallSiteVisitor visitor = new CallSiteVisitor();
|
||||
new JsonParser(new JsonVisitingConsumer(new JsonArrayVisitor(visitor))).parse(reader);
|
||||
visitor.flush();
|
||||
}
|
||||
|
||||
public Location[] getLocations(int callSiteId) {
|
||||
CallSite callSite = callSites.get(callSiteId);
|
||||
if (callSite == null) {
|
||||
return null;
|
||||
}
|
||||
|
||||
Location[] result = new Location[callSite.locations.length];
|
||||
for (int i = 0; i < result.length; ++i) {
|
||||
result[i] = callSite.locations[i].clone();
|
||||
}
|
||||
return result;
|
||||
}
|
||||
|
||||
public String deobfuscate(Reader text) throws IOException {
|
||||
StringBuilder result = new StringBuilder();
|
||||
|
||||
String expectedPrefix = " at Obfuscated.obfuscated(Obfuscated.java:";
|
||||
int expectedPrefixPos = 0;
|
||||
boolean isInNumber = false;
|
||||
boolean expectingLineEnd = false;
|
||||
StringBuilder sb = new StringBuilder();
|
||||
|
||||
while (true) {
|
||||
int c = text.read();
|
||||
|
||||
if (expectedPrefixPos >= 0) {
|
||||
if (expectedPrefix.charAt(expectedPrefixPos) == c) {
|
||||
expectedPrefixPos++;
|
||||
if (expectedPrefixPos == expectedPrefix.length()) {
|
||||
expectedPrefixPos = -1;
|
||||
isInNumber = true;
|
||||
}
|
||||
continue;
|
||||
}
|
||||
result.append(expectedPrefix, 0, expectedPrefixPos);
|
||||
expectedPrefixPos = -1;
|
||||
} else if (isInNumber) {
|
||||
if (c >= '0' && c <= '9') {
|
||||
sb.append((char) c);
|
||||
continue;
|
||||
} else if (c == ')') {
|
||||
isInNumber = false;
|
||||
expectingLineEnd = true;
|
||||
continue;
|
||||
}
|
||||
isInNumber = false;
|
||||
result.append(expectedPrefix).append(sb);
|
||||
}
|
||||
|
||||
if (c == '\r' || c == '\n' || c < 0) {
|
||||
if (expectingLineEnd) {
|
||||
int callSiteId = Integer.parseInt(sb.toString());
|
||||
Location[] locations = getLocations(callSiteId);
|
||||
if (locations == null) {
|
||||
result.append(expectedPrefix).append(sb).append(')');
|
||||
} else {
|
||||
boolean first = true;
|
||||
for (Location location : locations) {
|
||||
if (!first) {
|
||||
result.append('\n');
|
||||
}
|
||||
first = false;
|
||||
result.append(" ");
|
||||
result.append(location.className).append('.').append(location.methodName).append('(');
|
||||
if (location.fileName != null & location.line >= 0) {
|
||||
result.append(location.fileName).append(':').append(location.line);
|
||||
} else {
|
||||
result.append("Unknown location");
|
||||
}
|
||||
result.append(")");
|
||||
}
|
||||
}
|
||||
sb.setLength(0);
|
||||
}
|
||||
expectingLineEnd = false;
|
||||
expectedPrefixPos = 0;
|
||||
}
|
||||
if (c < 0) {
|
||||
break;
|
||||
}
|
||||
result.append((char) c);
|
||||
}
|
||||
|
||||
return result.toString();
|
||||
}
|
||||
|
||||
class CallSiteVisitor extends JsonAllErrorVisitor {
|
||||
CallSite callSite;
|
||||
List<Location> locations = new ArrayList<>();
|
||||
Location location;
|
||||
JsonPropertyVisitor propertyVisitor = new JsonPropertyVisitor(false);
|
||||
JsonPropertyVisitor locationPropertyVisitor = new JsonPropertyVisitor(false);
|
||||
|
||||
CallSiteVisitor() {
|
||||
propertyVisitor.addProperty("id", new JsonAllErrorVisitor() {
|
||||
@Override
|
||||
public void intValue(JsonErrorReporter reporter, long value) {
|
||||
callSite.id = (int) value;
|
||||
}
|
||||
});
|
||||
propertyVisitor.addProperty("locations", new JsonArrayVisitor(locationVisitor));
|
||||
|
||||
locationPropertyVisitor.addProperty("file", new JsonAllErrorVisitor() {
|
||||
@Override
|
||||
public void stringValue(JsonErrorReporter reporter, String value) {
|
||||
location.fileName = value;
|
||||
}
|
||||
|
||||
@Override
|
||||
public void nullValue(JsonErrorReporter reporter) {
|
||||
location.fileName = null;
|
||||
}
|
||||
});
|
||||
|
||||
locationPropertyVisitor.addProperty("class", new JsonAllErrorVisitor() {
|
||||
@Override
|
||||
public void stringValue(JsonErrorReporter reporter, String value) {
|
||||
location.className = value;
|
||||
}
|
||||
|
||||
@Override
|
||||
public void nullValue(JsonErrorReporter reporter) {
|
||||
location.className = null;
|
||||
}
|
||||
});
|
||||
|
||||
locationPropertyVisitor.addProperty("method", new JsonAllErrorVisitor() {
|
||||
@Override
|
||||
public void stringValue(JsonErrorReporter reporter, String value) {
|
||||
location.methodName = value;
|
||||
}
|
||||
|
||||
@Override
|
||||
public void nullValue(JsonErrorReporter reporter) {
|
||||
location.methodName = null;
|
||||
}
|
||||
});
|
||||
|
||||
locationPropertyVisitor.addProperty("line", new JsonAllErrorVisitor() {
|
||||
@Override
|
||||
public void intValue(JsonErrorReporter reporter, long value) {
|
||||
location.line = (int) value;
|
||||
}
|
||||
});
|
||||
}
|
||||
|
||||
@Override
|
||||
public JsonVisitor object(JsonErrorReporter reporter) {
|
||||
flush();
|
||||
callSite = new CallSite();
|
||||
return propertyVisitor;
|
||||
}
|
||||
|
||||
void flush() {
|
||||
if (callSite != null) {
|
||||
callSites.put(callSite.id, callSite);
|
||||
callSite.locations = locations.toArray(new Location[0]);
|
||||
locations.clear();
|
||||
}
|
||||
}
|
||||
|
||||
JsonAllErrorVisitor locationVisitor = new JsonAllErrorVisitor() {
|
||||
@Override
|
||||
public JsonVisitor object(JsonErrorReporter reporter) {
|
||||
location = new Location();
|
||||
locations.add(location);
|
||||
return locationPropertyVisitor;
|
||||
}
|
||||
};
|
||||
}
|
||||
|
||||
public static void main(String[] args) throws IOException {
|
||||
if (args.length != 2) {
|
||||
System.err.println("Two arguments expected (JSON table file, stack trace file)");
|
||||
System.exit(1);
|
||||
}
|
||||
|
||||
Deobfuscator deobfuscator;
|
||||
try (InputStream input = new BufferedInputStream(new FileInputStream(args[0]));
|
||||
Reader reader = new InputStreamReader(input, StandardCharsets.UTF_8)) {
|
||||
deobfuscator = new Deobfuscator(reader);
|
||||
}
|
||||
|
||||
String result;
|
||||
try (InputStream input = new BufferedInputStream(new FileInputStream(args[1]));
|
||||
Reader reader = new InputStreamReader(input, StandardCharsets.UTF_8)) {
|
||||
result = deobfuscator.deobfuscate(reader);
|
||||
}
|
||||
|
||||
System.out.println(result);
|
||||
}
|
||||
|
||||
static class CallSite {
|
||||
int id;
|
||||
Location[] locations;
|
||||
}
|
||||
|
||||
public static class Location implements Cloneable {
|
||||
public String className;
|
||||
public String methodName;
|
||||
public String fileName;
|
||||
public int line = -1;
|
||||
|
||||
@Override
|
||||
protected Location clone() {
|
||||
try {
|
||||
return (Location) super.clone();
|
||||
} catch (CloneNotSupportedException e) {
|
||||
throw new AssertionError(e);
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
Loading…
Reference in New Issue
Block a user