mirror of
https://github.com/Eaglercraft-TeaVM-Fork/eagler-teavm.git
synced 2025-01-08 07:54:11 -08:00
Multiple small improvements
This commit is contained in:
parent
1b374c7466
commit
569d6fa3e8
|
@ -62,30 +62,45 @@ public class ChromeRDPDebugger implements JavaScriptDebugger, ChromeRDPExchangeC
|
||||||
|
|
||||||
private ChromeRDPExchangeListener exchangeListener = new ChromeRDPExchangeListener() {
|
private ChromeRDPExchangeListener exchangeListener = new ChromeRDPExchangeListener() {
|
||||||
@Override public void received(String messageText) throws IOException {
|
@Override public void received(String messageText) throws IOException {
|
||||||
JsonNode jsonMessage = mapper.readTree(messageText);
|
receiveMessage(messageText);
|
||||||
if (jsonMessage.has("id")) {
|
|
||||||
Response response = mapper.reader(Response.class).readValue(jsonMessage);
|
|
||||||
responseHandlers.remove(response.getId()).received(response.getResult());
|
|
||||||
} else {
|
|
||||||
Message message = mapper.reader(Message.class).readValue(messageText);
|
|
||||||
if (message.getMethod() == null) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
switch (message.getMethod()) {
|
|
||||||
case "Debugger.paused":
|
|
||||||
firePaused(parseJson(SuspendedNotification.class, message.getParams()));
|
|
||||||
break;
|
|
||||||
case "Debugger.resumed":
|
|
||||||
fireResumed();
|
|
||||||
break;
|
|
||||||
case "Debugger.scriptParsed":
|
|
||||||
scriptParsed(parseJson(ScriptParsedNotification.class, message.getParams()));
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
private void receiveMessage(final String messageText) {
|
||||||
|
new Thread() {
|
||||||
|
@Override public void run() {
|
||||||
|
try {
|
||||||
|
JsonNode jsonMessage = mapper.readTree(messageText);
|
||||||
|
if (jsonMessage.has("id")) {
|
||||||
|
Response response = mapper.reader(Response.class).readValue(jsonMessage);
|
||||||
|
if (response.getError() != null) {
|
||||||
|
System.err.println(response.getError().toString());
|
||||||
|
}
|
||||||
|
responseHandlers.remove(response.getId()).received(response.getResult());
|
||||||
|
} else {
|
||||||
|
Message message = mapper.reader(Message.class).readValue(messageText);
|
||||||
|
if (message.getMethod() == null) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
switch (message.getMethod()) {
|
||||||
|
case "Debugger.paused":
|
||||||
|
firePaused(parseJson(SuspendedNotification.class, message.getParams()));
|
||||||
|
break;
|
||||||
|
case "Debugger.resumed":
|
||||||
|
fireResumed();
|
||||||
|
break;
|
||||||
|
case "Debugger.scriptParsed":
|
||||||
|
scriptParsed(parseJson(ScriptParsedNotification.class, message.getParams()));
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} catch (Exception e) {
|
||||||
|
// TODO: logging
|
||||||
|
e.printStackTrace();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}.start();
|
||||||
|
}
|
||||||
|
|
||||||
private synchronized void firePaused(SuspendedNotification params) {
|
private synchronized void firePaused(SuspendedNotification params) {
|
||||||
suspended = true;
|
suspended = true;
|
||||||
|
@ -256,8 +271,12 @@ public class ChromeRDPDebugger implements JavaScriptDebugger, ChromeRDPExchangeC
|
||||||
message.setParams(mapper.valueToTree(params));
|
message.setParams(mapper.valueToTree(params));
|
||||||
ResponseHandler handler = new ResponseHandler() {
|
ResponseHandler handler = new ResponseHandler() {
|
||||||
@Override public void received(JsonNode node) throws IOException {
|
@Override public void received(JsonNode node) throws IOException {
|
||||||
SetBreakpointResponse response = mapper.reader(SetBreakpointResponse.class).readValue(node);
|
if (node != null) {
|
||||||
breakpoint.chromeId = response.getBreakpointId();
|
SetBreakpointResponse response = mapper.reader(SetBreakpointResponse.class).readValue(node);
|
||||||
|
breakpoint.chromeId = response.getBreakpointId();
|
||||||
|
} else {
|
||||||
|
breakpoint.chromeId = null;
|
||||||
|
}
|
||||||
for (JavaScriptDebuggerListener listener : getListeners()) {
|
for (JavaScriptDebuggerListener listener : getListeners()) {
|
||||||
listener.breakpointChanged(breakpoint);
|
listener.breakpointChanged(breakpoint);
|
||||||
}
|
}
|
||||||
|
@ -314,7 +333,8 @@ public class ChromeRDPDebugger implements JavaScriptDebugger, ChromeRDPExchangeC
|
||||||
sync.add("");
|
sync.add("");
|
||||||
} else {
|
} else {
|
||||||
CallFunctionResponse response = mapper.reader(CallFunctionResponse.class).readValue(node);
|
CallFunctionResponse response = mapper.reader(CallFunctionResponse.class).readValue(node);
|
||||||
sync.add(response.getResult().getValue().getTextValue());
|
RemoteObjectDTO result = response.getResult();
|
||||||
|
sync.add(result.getValue() != null ? result.getValue().getTextValue() : "");
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
@ -327,6 +347,51 @@ public class ChromeRDPDebugger implements JavaScriptDebugger, ChromeRDPExchangeC
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
String getRepresentation(String objectId) {
|
||||||
|
if (exchange == null) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
Message message = new Message();
|
||||||
|
message.setId(messageIdGenerator.incrementAndGet());
|
||||||
|
message.setMethod("Runtime.callFunctionOn");
|
||||||
|
CallFunctionCommand params = new CallFunctionCommand();
|
||||||
|
CallArgumentDTO arg = new CallArgumentDTO();
|
||||||
|
arg.setObjectId(objectId);
|
||||||
|
params.setObjectId(objectId);
|
||||||
|
params.setArguments(new CallArgumentDTO[] { arg });
|
||||||
|
params.setFunctionDeclaration("$dbg_repr");
|
||||||
|
message.setParams(mapper.valueToTree(params));
|
||||||
|
final BlockingQueue<RepresentationWrapper> sync = new LinkedTransferQueue<>();
|
||||||
|
responseHandlers.put(message.getId(), new ResponseHandler() {
|
||||||
|
@Override public void received(JsonNode node) throws IOException {
|
||||||
|
if (node == null) {
|
||||||
|
sync.add(new RepresentationWrapper(null));
|
||||||
|
} else {
|
||||||
|
CallFunctionResponse response = mapper.reader(CallFunctionResponse.class).readValue(node);
|
||||||
|
RemoteObjectDTO result = response.getResult();
|
||||||
|
sync.add(new RepresentationWrapper(result.getValue() != null ?
|
||||||
|
result.getValue().getTextValue() : null));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
sendMessage(message);
|
||||||
|
try {
|
||||||
|
RepresentationWrapper result = sync.take();
|
||||||
|
return result.repr;
|
||||||
|
} catch (InterruptedException e) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private static class RepresentationWrapper {
|
||||||
|
String repr;
|
||||||
|
|
||||||
|
public RepresentationWrapper(String repr) {
|
||||||
|
super();
|
||||||
|
this.repr = repr;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
private List<RDPLocalVariable> parseProperties(PropertyDescriptorDTO[] properties) {
|
private List<RDPLocalVariable> parseProperties(PropertyDescriptorDTO[] properties) {
|
||||||
List<RDPLocalVariable> variables = new ArrayList<>();
|
List<RDPLocalVariable> variables = new ArrayList<>();
|
||||||
if (properties != null) {
|
if (properties != null) {
|
||||||
|
@ -339,8 +404,7 @@ public class ChromeRDPDebugger implements JavaScriptDebugger, ChromeRDPExchangeC
|
||||||
break;
|
break;
|
||||||
case "object":
|
case "object":
|
||||||
case "function":
|
case "function":
|
||||||
value = new RDPValue(this, remoteValue.getDescription(), remoteValue.getType(),
|
value = new RDPValue(this, null, remoteValue.getType(), remoteValue.getObjectId());
|
||||||
remoteValue.getObjectId());
|
|
||||||
break;
|
break;
|
||||||
default:
|
default:
|
||||||
value = new RDPValue(this, remoteValue.getValue().asText(), remoteValue.getType(),
|
value = new RDPValue(this, remoteValue.getValue().asText(), remoteValue.getType(),
|
||||||
|
|
|
@ -2,6 +2,7 @@ package org.teavm.chromerdp;
|
||||||
|
|
||||||
import java.util.Collections;
|
import java.util.Collections;
|
||||||
import java.util.Map;
|
import java.util.Map;
|
||||||
|
import java.util.concurrent.atomic.AtomicReference;
|
||||||
import org.teavm.debugging.JavaScriptValue;
|
import org.teavm.debugging.JavaScriptValue;
|
||||||
import org.teavm.debugging.JavaScriptVariable;
|
import org.teavm.debugging.JavaScriptVariable;
|
||||||
|
|
||||||
|
@ -10,14 +11,15 @@ import org.teavm.debugging.JavaScriptVariable;
|
||||||
* @author Alexey Andreev <konsoletyper@gmail.com>
|
* @author Alexey Andreev <konsoletyper@gmail.com>
|
||||||
*/
|
*/
|
||||||
public class RDPValue implements JavaScriptValue {
|
public class RDPValue implements JavaScriptValue {
|
||||||
private String representation;
|
private AtomicReference<String> representation = new AtomicReference<>();
|
||||||
|
private AtomicReference<String> className = new AtomicReference<>();
|
||||||
private String typeName;
|
private String typeName;
|
||||||
private ChromeRDPDebugger debugger;
|
private ChromeRDPDebugger debugger;
|
||||||
private String objectId;
|
private String objectId;
|
||||||
private Map<String, ? extends JavaScriptVariable> properties;
|
private Map<String, ? extends JavaScriptVariable> properties;
|
||||||
|
|
||||||
public RDPValue(ChromeRDPDebugger debugger, String representation, String typeName, String objectId) {
|
public RDPValue(ChromeRDPDebugger debugger, String representation, String typeName, String objectId) {
|
||||||
this.representation = representation;
|
this.representation.set(representation == null && objectId == null ? "" : representation);
|
||||||
this.typeName = typeName;
|
this.typeName = typeName;
|
||||||
this.debugger = debugger;
|
this.debugger = debugger;
|
||||||
this.objectId = objectId;
|
this.objectId = objectId;
|
||||||
|
@ -27,17 +29,23 @@ public class RDPValue implements JavaScriptValue {
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public String getRepresentation() {
|
public String getRepresentation() {
|
||||||
return representation;
|
if (representation.get() == null) {
|
||||||
|
representation.compareAndSet(null, debugger.getRepresentation(objectId));
|
||||||
|
}
|
||||||
|
return representation.get();
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public String getClassName() {
|
public String getClassName() {
|
||||||
if (objectId != null) {
|
if (className.get() == null) {
|
||||||
String className = debugger.getClassName(objectId);
|
if (objectId != null) {
|
||||||
return className != null ? className : "object";
|
String computedClassName = debugger.getClassName(objectId);
|
||||||
} else {
|
className.compareAndSet(null, computedClassName != null ? computedClassName : "@Object");
|
||||||
return typeName;
|
} else {
|
||||||
|
className.compareAndSet(null, "@" + typeName);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
return className.get();
|
||||||
}
|
}
|
||||||
|
|
||||||
@SuppressWarnings("unchecked")
|
@SuppressWarnings("unchecked")
|
||||||
|
|
|
@ -26,6 +26,7 @@ import org.codehaus.jackson.annotate.JsonIgnoreProperties;
|
||||||
public class Response {
|
public class Response {
|
||||||
private int id;
|
private int id;
|
||||||
private JsonNode result;
|
private JsonNode result;
|
||||||
|
private JsonNode error;
|
||||||
|
|
||||||
public int getId() {
|
public int getId() {
|
||||||
return id;
|
return id;
|
||||||
|
@ -42,4 +43,12 @@ public class Response {
|
||||||
public void setResult(JsonNode result) {
|
public void setResult(JsonNode result) {
|
||||||
this.result = result;
|
this.result = result;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public JsonNode getError() {
|
||||||
|
return error;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setError(JsonNode error) {
|
||||||
|
this.error = error;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -33,6 +33,7 @@ public class JCLPlugin implements TeaVMPlugin {
|
||||||
host.add(new EnumTransformer());
|
host.add(new EnumTransformer());
|
||||||
host.add(new ClassLookupDependencySupport());
|
host.add(new ClassLookupDependencySupport());
|
||||||
host.add(new NewInstanceDependencySupport());
|
host.add(new NewInstanceDependencySupport());
|
||||||
|
host.add(new ObjectEnrichRenderer());
|
||||||
ServiceLoaderSupport serviceLoaderSupp = new ServiceLoaderSupport(host.getClassLoader());
|
ServiceLoaderSupport serviceLoaderSupp = new ServiceLoaderSupport(host.getClassLoader());
|
||||||
host.add(serviceLoaderSupp);
|
host.add(serviceLoaderSupp);
|
||||||
MethodReference loadServicesMethod = new MethodReference("java.util.ServiceLoader", new MethodDescriptor(
|
MethodReference loadServicesMethod = new MethodReference("java.util.ServiceLoader", new MethodDescriptor(
|
||||||
|
|
|
@ -0,0 +1,62 @@
|
||||||
|
/*
|
||||||
|
* Copyright 2014 Alexey Andreev.
|
||||||
|
*
|
||||||
|
* 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.classlib.impl;
|
||||||
|
|
||||||
|
import java.io.IOException;
|
||||||
|
import org.teavm.javascript.RenderingContext;
|
||||||
|
import org.teavm.model.ClassReader;
|
||||||
|
import org.teavm.model.MethodDescriptor;
|
||||||
|
import org.teavm.model.MethodReader;
|
||||||
|
import org.teavm.vm.BuildTarget;
|
||||||
|
import org.teavm.vm.spi.RendererListener;
|
||||||
|
|
||||||
|
/**
|
||||||
|
*
|
||||||
|
* @author Alexey Andreev
|
||||||
|
*/
|
||||||
|
public class ObjectEnrichRenderer implements RendererListener {
|
||||||
|
private RenderingContext context;
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void begin(RenderingContext context, BuildTarget buildTarget) throws IOException {
|
||||||
|
this.context = context;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void beforeClass(ClassReader cls) throws IOException {
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void afterClass(ClassReader cls) throws IOException {
|
||||||
|
if (cls.getName().equals("java.lang.Object")) {
|
||||||
|
MethodReader toString = cls.getMethod(new MethodDescriptor("toString", String.class));
|
||||||
|
if (toString != null) {
|
||||||
|
String clsName = context.getNaming().getNameFor(cls.getName());
|
||||||
|
String toStringName = context.getNaming().getNameFor(toString.getReference());
|
||||||
|
context.getWriter().append(clsName).append(".prototype.toString").ws().append('=').ws()
|
||||||
|
.append("function()").ws().append('{').indent().softNewLine();
|
||||||
|
context.getWriter().append("return this.").append(toStringName).ws().append('?').ws()
|
||||||
|
.append("$rt_ustr(this.").append(toStringName).append("())").ws().append(':')
|
||||||
|
.append("Object.prototype.toString.call(this);").softNewLine();
|
||||||
|
context.getWriter().outdent().append("}").newLine();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void complete() throws IOException {
|
||||||
|
}
|
||||||
|
}
|
|
@ -45,7 +45,7 @@ public class DebugInformation {
|
||||||
Mapping methodMapping;
|
Mapping methodMapping;
|
||||||
Mapping lineMapping;
|
Mapping lineMapping;
|
||||||
MultiMapping[] variableMappings;
|
MultiMapping[] variableMappings;
|
||||||
List<Map<Integer, Integer>> classesMetadata;
|
List<ClassMetadata> classesMetadata;
|
||||||
|
|
||||||
public String[] getCoveredSourceFiles() {
|
public String[] getCoveredSourceFiles() {
|
||||||
return fileNames.clone();
|
return fileNames.clone();
|
||||||
|
@ -130,8 +130,15 @@ public class DebugInformation {
|
||||||
if (jsIndex == null) {
|
if (jsIndex == null) {
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
Integer fieldIndex = classesMetadata.get(classIndex).get(jsIndex);
|
while (classIndex != null) {
|
||||||
return fieldIndex != null ? fields[fieldIndex] : null;
|
ClassMetadata cls = classesMetadata.get(classIndex);
|
||||||
|
Integer fieldIndex = cls.fieldMap.get(jsIndex);
|
||||||
|
if (fieldIndex != null) {
|
||||||
|
return fields[fieldIndex];
|
||||||
|
}
|
||||||
|
classIndex = cls.parentId;
|
||||||
|
}
|
||||||
|
return null;
|
||||||
}
|
}
|
||||||
|
|
||||||
private <T> T componentByKey(Mapping mapping, T[] values, GeneratedLocation location) {
|
private <T> T componentByKey(Mapping mapping, T[] values, GeneratedLocation location) {
|
||||||
|
@ -372,4 +379,9 @@ public class DebugInformation {
|
||||||
return lines.length;
|
return lines.length;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static class ClassMetadata {
|
||||||
|
Integer parentId;
|
||||||
|
Map<Integer, Integer> fieldMap = new HashMap<>();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -104,12 +104,14 @@ public class DebugInformationBuilder implements DebugInformationEmitter {
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void addClass(String className) {
|
public void addClass(String className, String parentName) {
|
||||||
int classIndex = classes.index(className);
|
int classIndex = classes.index(className);
|
||||||
|
int parentIndex = classes.index(parentName);
|
||||||
while (classIndex >= classesMetadata.size()) {
|
while (classIndex >= classesMetadata.size()) {
|
||||||
classesMetadata.add(new ClassMetadata());
|
classesMetadata.add(new ClassMetadata());
|
||||||
}
|
}
|
||||||
currentClassMetadata = classIndex;
|
currentClassMetadata = classIndex;
|
||||||
|
classesMetadata.get(currentClassMetadata).parentIndex = parentIndex;
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
|
@ -140,13 +142,16 @@ public class DebugInformationBuilder implements DebugInformationEmitter {
|
||||||
debugInformation.variableMappings[var] = mapping.build();
|
debugInformation.variableMappings[var] = mapping.build();
|
||||||
}
|
}
|
||||||
|
|
||||||
List<Map<Integer, Integer>> builtMetadata = new ArrayList<>(classes.list.size());
|
List<DebugInformation.ClassMetadata> builtMetadata = new ArrayList<>(classes.list.size());
|
||||||
for (int i = 0; i < classes.list.size(); ++i) {
|
for (int i = 0; i < classes.list.size(); ++i) {
|
||||||
if (i >= classesMetadata.size()) {
|
if (i >= classesMetadata.size()) {
|
||||||
builtMetadata.add(new HashMap<Integer, Integer>());
|
builtMetadata.add(new DebugInformation.ClassMetadata());
|
||||||
} else {
|
} else {
|
||||||
Map<Integer, Integer> map = new HashMap<>(classesMetadata.get(i).fieldMap);
|
ClassMetadata origMetadata = classesMetadata.get(i);
|
||||||
builtMetadata.add(map);
|
DebugInformation.ClassMetadata mappedMetadata = new DebugInformation.ClassMetadata();
|
||||||
|
mappedMetadata.fieldMap.putAll(origMetadata.fieldMap);
|
||||||
|
mappedMetadata.parentId = origMetadata.parentIndex >= 0 ? origMetadata.parentIndex : null;
|
||||||
|
builtMetadata.add(mappedMetadata);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
debugInformation.classesMetadata = builtMetadata;
|
debugInformation.classesMetadata = builtMetadata;
|
||||||
|
@ -285,6 +290,7 @@ public class DebugInformationBuilder implements DebugInformationEmitter {
|
||||||
}
|
}
|
||||||
|
|
||||||
static class ClassMetadata {
|
static class ClassMetadata {
|
||||||
|
int parentIndex;
|
||||||
Map<Integer, Integer> fieldMap = new HashMap<>();
|
Map<Integer, Integer> fieldMap = new HashMap<>();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -33,7 +33,7 @@ public interface DebugInformationEmitter {
|
||||||
|
|
||||||
void emitVariable(String[] sourceNames, String generatedName);
|
void emitVariable(String[] sourceNames, String generatedName);
|
||||||
|
|
||||||
void addClass(String className);
|
void addClass(String className, String parentName);
|
||||||
|
|
||||||
void addField(String fieldName, String jsName);
|
void addField(String fieldName, String jsName);
|
||||||
}
|
}
|
|
@ -19,9 +19,7 @@ import java.io.EOFException;
|
||||||
import java.io.IOException;
|
import java.io.IOException;
|
||||||
import java.io.InputStream;
|
import java.io.InputStream;
|
||||||
import java.util.ArrayList;
|
import java.util.ArrayList;
|
||||||
import java.util.HashMap;
|
|
||||||
import java.util.List;
|
import java.util.List;
|
||||||
import java.util.Map;
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
*
|
*
|
||||||
|
@ -64,17 +62,21 @@ class DebugInformationReader {
|
||||||
return mappings;
|
return mappings;
|
||||||
}
|
}
|
||||||
|
|
||||||
private List<Map<Integer, Integer>> readClassesMetadata(int count) throws IOException {
|
private List<DebugInformation.ClassMetadata> readClassesMetadata(int count) throws IOException {
|
||||||
List<Map<Integer, Integer>> classes = new ArrayList<>(count);
|
List<DebugInformation.ClassMetadata> classes = new ArrayList<>(count);
|
||||||
for (int i = 0; i < count; ++i) {
|
for (int i = 0; i < count; ++i) {
|
||||||
Map<Integer, Integer> cls = new HashMap<>();
|
DebugInformation.ClassMetadata cls = new DebugInformation.ClassMetadata();
|
||||||
classes.add(cls);
|
classes.add(cls);
|
||||||
|
cls.parentId = readUnsignedNumber() - 1;
|
||||||
|
if (cls.parentId.equals(-1)) {
|
||||||
|
cls.parentId = null;
|
||||||
|
}
|
||||||
int entryCount = readUnsignedNumber();
|
int entryCount = readUnsignedNumber();
|
||||||
resetRelativeNumber();
|
resetRelativeNumber();
|
||||||
for (int j = 0; j < entryCount; ++j) {
|
for (int j = 0; j < entryCount; ++j) {
|
||||||
int key = readRelativeNumber();
|
int key = readRelativeNumber();
|
||||||
int value = readUnsignedNumber();
|
int value = readUnsignedNumber();
|
||||||
cls.put(key, value);
|
cls.fieldMap.put(key, value);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return classes;
|
return classes;
|
||||||
|
|
|
@ -20,7 +20,7 @@ import java.io.IOException;
|
||||||
import java.util.ArrayList;
|
import java.util.ArrayList;
|
||||||
import java.util.Collections;
|
import java.util.Collections;
|
||||||
import java.util.List;
|
import java.util.List;
|
||||||
import java.util.Map;
|
import org.teavm.debugging.DebugInformation.ClassMetadata;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
*
|
*
|
||||||
|
@ -63,16 +63,17 @@ class DebugInformationWriter {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private void writeClassMetadata(List<Map<Integer, Integer>> classes) throws IOException {
|
private void writeClassMetadata(List<ClassMetadata> classes) throws IOException {
|
||||||
for (int i = 0; i < classes.size(); ++i) {
|
for (int i = 0; i < classes.size(); ++i) {
|
||||||
Map<Integer, Integer> cls = classes.get(i);
|
ClassMetadata cls = classes.get(i);
|
||||||
writeUnsignedNumber(cls.size());
|
writeUnsignedNumber(cls.parentId != null ? cls.parentId + 1 : 0);
|
||||||
List<Integer> keys = new ArrayList<>(cls.keySet());
|
writeUnsignedNumber(cls.fieldMap.size());
|
||||||
|
List<Integer> keys = new ArrayList<>(cls.fieldMap.keySet());
|
||||||
Collections.sort(keys);
|
Collections.sort(keys);
|
||||||
resetRelativeNumber();
|
resetRelativeNumber();
|
||||||
for (int key : keys) {
|
for (int key : keys) {
|
||||||
writeRelativeNumber(key);
|
writeRelativeNumber(key);
|
||||||
writeUnsignedNumber(cls.get(key));
|
writeUnsignedNumber(cls.fieldMap.get(key));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -26,8 +26,6 @@ import org.teavm.model.MethodReference;
|
||||||
*
|
*
|
||||||
* @author Alexey Andreev
|
* @author Alexey Andreev
|
||||||
*/
|
*/
|
||||||
// TODO: variable name handling
|
|
||||||
// TODO: class fields handling
|
|
||||||
public class Debugger {
|
public class Debugger {
|
||||||
private static final Object dummyObject = new Object();
|
private static final Object dummyObject = new Object();
|
||||||
private ConcurrentMap<DebuggerListener, Object> listeners = new ConcurrentHashMap<>();
|
private ConcurrentMap<DebuggerListener, Object> listeners = new ConcurrentHashMap<>();
|
||||||
|
@ -198,13 +196,16 @@ public class Debugger {
|
||||||
|
|
||||||
private void addScript(String name) {
|
private void addScript(String name) {
|
||||||
if (debugInformationMap.containsKey(name)) {
|
if (debugInformationMap.containsKey(name)) {
|
||||||
|
updateBreakpoints();
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
DebugInformation debugInfo = debugInformationProvider.getDebugInformation(name);
|
DebugInformation debugInfo = debugInformationProvider.getDebugInformation(name);
|
||||||
if (debugInfo == null) {
|
if (debugInfo == null) {
|
||||||
|
updateBreakpoints();
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
if (debugInformationMap.putIfAbsent(name, debugInfo) != null) {
|
if (debugInformationMap.putIfAbsent(name, debugInfo) != null) {
|
||||||
|
updateBreakpoints();
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
for (String sourceFile : debugInfo.getCoveredSourceFiles()) {
|
for (String sourceFile : debugInfo.getCoveredSourceFiles()) {
|
||||||
|
@ -220,6 +221,10 @@ public class Debugger {
|
||||||
list.put(debugInfo, dummyObject);
|
list.put(debugInfo, dummyObject);
|
||||||
}
|
}
|
||||||
scriptMap.put(debugInfo, name);
|
scriptMap.put(debugInfo, name);
|
||||||
|
updateBreakpoints();
|
||||||
|
}
|
||||||
|
|
||||||
|
private void updateBreakpoints() {
|
||||||
for (Breakpoint breakpoint : breakpoints.keySet()) {
|
for (Breakpoint breakpoint : breakpoints.keySet()) {
|
||||||
updateInternalBreakpoints(breakpoint);
|
updateInternalBreakpoints(breakpoint);
|
||||||
updateBreakpointStatus(breakpoint, true);
|
updateBreakpointStatus(breakpoint, true);
|
||||||
|
|
|
@ -45,7 +45,7 @@ public class DummyDebugInformationEmitter implements DebugInformationEmitter {
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void addClass(String className) {
|
public void addClass(String className, String parentName) {
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
|
|
|
@ -243,7 +243,7 @@ public class Renderer implements ExprVisitor, StatementVisitor, RenderingContext
|
||||||
|
|
||||||
public void render(ClassNode cls) throws RenderingException {
|
public void render(ClassNode cls) throws RenderingException {
|
||||||
debugEmitter.emitClass(cls.getName());
|
debugEmitter.emitClass(cls.getName());
|
||||||
debugEmitter.addClass(cls.getName());
|
debugEmitter.addClass(cls.getName(), cls.getParentName());
|
||||||
try {
|
try {
|
||||||
writer.append("function ").appendClass(cls.getName()).append("()").ws().append("{")
|
writer.append("function ").appendClass(cls.getName()).append("()").ws().append("{")
|
||||||
.indent().softNewLine();
|
.indent().softNewLine();
|
||||||
|
|
|
@ -108,6 +108,17 @@ $rt_arraycls = function(cls) {
|
||||||
};
|
};
|
||||||
arraycls.prototype = new ($rt_objcls())();
|
arraycls.prototype = new ($rt_objcls())();
|
||||||
arraycls.prototype.constructor = arraycls;
|
arraycls.prototype.constructor = arraycls;
|
||||||
|
arraycls.prototype.toString = function() {
|
||||||
|
var str = "[";
|
||||||
|
for (var i = 0; i < this.data.length; ++i) {
|
||||||
|
if (i > 0) {
|
||||||
|
str += ", ";
|
||||||
|
}
|
||||||
|
str += this.data[i].toString();
|
||||||
|
}
|
||||||
|
str += "]";
|
||||||
|
return str;
|
||||||
|
}
|
||||||
arraycls.$meta = { item : cls, supertypes : [$rt_objcls()], primitive : false, superclass : $rt_objcls() };
|
arraycls.$meta = { item : cls, supertypes : [$rt_objcls()], primitive : false, superclass : $rt_objcls() };
|
||||||
cls.$array = arraycls;
|
cls.$array = arraycls;
|
||||||
}
|
}
|
||||||
|
@ -392,20 +403,64 @@ function $rt_s(index) {
|
||||||
}
|
}
|
||||||
|
|
||||||
function $dbg_repr(obj) {
|
function $dbg_repr(obj) {
|
||||||
return obj.toString();
|
return obj.toString ? obj.toString() : "";
|
||||||
}
|
}
|
||||||
function $dbg_class(obj) {
|
function $dbg_class(obj) {
|
||||||
if (obj instanceof Long) {
|
if (obj instanceof Long) {
|
||||||
return "long";
|
return "long";
|
||||||
}
|
}
|
||||||
var meta = obj.constructor.$meta;
|
var cls = obj.constructor;
|
||||||
return meta ? meta.name : "";
|
var arrayDegree = 0;
|
||||||
|
while (cls.$meta && cls.$meta.item) {
|
||||||
|
++arrayDegree;
|
||||||
|
cls = cls.$meta.item;
|
||||||
|
}
|
||||||
|
var clsName = "";
|
||||||
|
if (cls === $rt_booleancls()) {
|
||||||
|
clsName = "boolean";
|
||||||
|
} else if (cls === $rt_bytecls()) {
|
||||||
|
clsName = "byte";
|
||||||
|
} else if (cls === $rt_shortcls()) {
|
||||||
|
clsName = "short";
|
||||||
|
} else if (cls === $rt_charcls()) {
|
||||||
|
clsName = "char";
|
||||||
|
} else if (cls === $rt_intcls()) {
|
||||||
|
clsName = "int";
|
||||||
|
} else if (cls === $rt_longcls()) {
|
||||||
|
clsName = "long";
|
||||||
|
} else if (cls === $rt_floatcls()) {
|
||||||
|
clsName = "float";
|
||||||
|
} else if (cls === $rt_doublecls()) {
|
||||||
|
clsName = "double";
|
||||||
|
} else {
|
||||||
|
clsName = cls.$meta ? cls.$meta.name : "@" + cls.name;
|
||||||
|
}
|
||||||
|
while (arrayDegree-- > 0) {
|
||||||
|
clsName += "[]";
|
||||||
|
}
|
||||||
|
return clsName;
|
||||||
}
|
}
|
||||||
|
|
||||||
Long = function(lo, hi) {
|
Long = function(lo, hi) {
|
||||||
this.lo = lo | 0;
|
this.lo = lo | 0;
|
||||||
this.hi = hi | 0;
|
this.hi = hi | 0;
|
||||||
}
|
}
|
||||||
|
Long.prototype.toString = function() {
|
||||||
|
var result = [];
|
||||||
|
var n = this;
|
||||||
|
var positive = Long_isPositive(n);
|
||||||
|
if (!positive) {
|
||||||
|
n = Long_neg(n);
|
||||||
|
}
|
||||||
|
var radix = new Long(10, 0);
|
||||||
|
do {
|
||||||
|
var divRem = Long_divRem(n, radix);
|
||||||
|
result.push(String.fromCharCode(48 + divRem[1].lo));
|
||||||
|
n = divRem[0];
|
||||||
|
} while (n.lo != 0 || n.hi != 0);
|
||||||
|
result = result.reverse().join('');
|
||||||
|
return positive ? result : "-" + result;
|
||||||
|
}
|
||||||
Long_ZERO = new Long(0, 0);
|
Long_ZERO = new Long(0, 0);
|
||||||
Long_fromInt = function(val) {
|
Long_fromInt = function(val) {
|
||||||
return val >= 0 ? new Long(val, 0) : new Long(val, -1);
|
return val >= 0 ? new Long(val, 0) : new Long(val, -1);
|
||||||
|
|
Loading…
Reference in New Issue
Block a user