From 646c8ec4884724ea8c66d8faef0f455934582dc6 Mon Sep 17 00:00:00 2001 From: Alexey Andreev Date: Wed, 23 Nov 2022 11:20:07 +0100 Subject: [PATCH] Wasm: fix issue with forward references in DWARF, generate class references as pointer types --- .../backend/wasm/dwarf/DwarfConstants.java | 1 + .../backend/wasm/dwarf/DwarfPlaceholder.java | 2 +- .../wasm/generate/DwarfClassGenerator.java | 28 ++++++++++++++++++- 3 files changed, 29 insertions(+), 2 deletions(-) diff --git a/core/src/main/java/org/teavm/backend/wasm/dwarf/DwarfConstants.java b/core/src/main/java/org/teavm/backend/wasm/dwarf/DwarfConstants.java index 103f4d974..8f100efb3 100644 --- a/core/src/main/java/org/teavm/backend/wasm/dwarf/DwarfConstants.java +++ b/core/src/main/java/org/teavm/backend/wasm/dwarf/DwarfConstants.java @@ -22,6 +22,7 @@ public final class DwarfConstants { public static final int DW_TAG_CLASS_TYPE = 0x02; public static final int DW_TAG_FORMAL_PARAMETER = 0x05; + public static final int DW_TAG_POINTER_TYPE = 0x0F; public static final int DW_TAG_COMPILE_UNIT = 0x11; public static final int DW_TAG_BASE_TYPE = 0x24; public static final int DW_TAG_SUBPROGRAM = 0x2E; diff --git a/core/src/main/java/org/teavm/backend/wasm/dwarf/DwarfPlaceholder.java b/core/src/main/java/org/teavm/backend/wasm/dwarf/DwarfPlaceholder.java index 020fded80..029e566b8 100644 --- a/core/src/main/java/org/teavm/backend/wasm/dwarf/DwarfPlaceholder.java +++ b/core/src/main/java/org/teavm/backend/wasm/dwarf/DwarfPlaceholder.java @@ -31,8 +31,8 @@ public class DwarfPlaceholder { void addForwardRef(DwarfPlaceholderWriter writer, Marker marker) { if (forwardReferences == null) { forwardReferences = new ArrayList<>(); - forwardReferences.add(new ForwardRef(writer, marker)); } + forwardReferences.add(new ForwardRef(writer, marker)); } static class ForwardRef { diff --git a/core/src/main/java/org/teavm/backend/wasm/generate/DwarfClassGenerator.java b/core/src/main/java/org/teavm/backend/wasm/generate/DwarfClassGenerator.java index 6c4bc6a86..4d5bdcf9e 100644 --- a/core/src/main/java/org/teavm/backend/wasm/generate/DwarfClassGenerator.java +++ b/core/src/main/java/org/teavm/backend/wasm/generate/DwarfClassGenerator.java @@ -23,12 +23,15 @@ import static org.teavm.backend.wasm.dwarf.DwarfConstants.DW_AT_BYTE_SIZE; import static org.teavm.backend.wasm.dwarf.DwarfConstants.DW_AT_DECLARATION; import static org.teavm.backend.wasm.dwarf.DwarfConstants.DW_AT_ENCODING; import static org.teavm.backend.wasm.dwarf.DwarfConstants.DW_AT_NAME; +import static org.teavm.backend.wasm.dwarf.DwarfConstants.DW_AT_TYPE; import static org.teavm.backend.wasm.dwarf.DwarfConstants.DW_FORM_DATA1; import static org.teavm.backend.wasm.dwarf.DwarfConstants.DW_FORM_FLAG_PRESENT; +import static org.teavm.backend.wasm.dwarf.DwarfConstants.DW_FORM_REF4; import static org.teavm.backend.wasm.dwarf.DwarfConstants.DW_FORM_STRP; import static org.teavm.backend.wasm.dwarf.DwarfConstants.DW_TAG_BASE_TYPE; import static org.teavm.backend.wasm.dwarf.DwarfConstants.DW_TAG_CLASS_TYPE; import static org.teavm.backend.wasm.dwarf.DwarfConstants.DW_TAG_NAMESPACE; +import static org.teavm.backend.wasm.dwarf.DwarfConstants.DW_TAG_POINTER_TYPE; import static org.teavm.backend.wasm.dwarf.DwarfConstants.DW_TAG_SUBPROGRAM; import static org.teavm.backend.wasm.dwarf.DwarfConstants.DW_TAG_UNSPECIFIED_TYPE; import java.util.ArrayList; @@ -39,6 +42,7 @@ import java.util.Map; import org.teavm.backend.wasm.dwarf.DwarfAbbreviation; import org.teavm.backend.wasm.dwarf.DwarfInfoWriter; import org.teavm.backend.wasm.dwarf.DwarfPlaceholder; +import org.teavm.backend.wasm.dwarf.blob.Blob; import org.teavm.model.MethodDescriptor; import org.teavm.model.PrimitiveType; import org.teavm.model.ValueType; @@ -57,6 +61,7 @@ public class DwarfClassGenerator { private DwarfPlaceholder[] primitiveTypes = new DwarfPlaceholder[PrimitiveType.values().length]; private DwarfPlaceholder unspecifiedType; private DwarfAbbreviation baseTypeAbbrev; + private DwarfAbbreviation pointerAbbrev; private List postponedWrites = new ArrayList<>(); public DwarfClassGenerator(DwarfInfoWriter writer, DwarfStrings strings) { @@ -164,7 +169,7 @@ public class DwarfClassGenerator { } private DwarfPlaceholder getClassType(String name) { - return getClass(name).ptr; + return getClass(name).getPointerPtr(); } private DwarfPlaceholder getPrimitivePtr(ValueType.Primitive type) { @@ -242,6 +247,15 @@ public class DwarfClassGenerator { return baseTypeAbbrev; } + private DwarfAbbreviation getPointerAbbrev() { + if (pointerAbbrev == null) { + pointerAbbrev = writer.abbreviation(DW_TAG_POINTER_TYPE, false, blob -> { + blob.writeLEB(DW_AT_TYPE).writeLEB(DW_FORM_REF4); + }); + } + return pointerAbbrev; + } + public class Namespace { public final String name; final Map namespaces = new LinkedHashMap<>(); @@ -279,6 +293,7 @@ public class DwarfClassGenerator { public class ClassType { public final String name; final DwarfPlaceholder ptr; + private DwarfPlaceholder pointerPtr; final Map subprograms = new LinkedHashMap<>(); private ClassType(String name) { @@ -290,12 +305,23 @@ public class DwarfClassGenerator { return subprograms.computeIfAbsent(desc, d -> new Subprogram(d.getName(), desc)); } + public DwarfPlaceholder getPointerPtr() { + if (pointerPtr == null) { + pointerPtr = writer.placeholder(4); + } + return pointerPtr; + } + private void write() { writer.mark(ptr).tag(getClassTypeAbbrev()); writer.writeInt(strings.stringRef(name)); for (var child : subprograms.values()) { child.write(); } + if (pointerPtr != null) { + writer.mark(pointerPtr).tag(getPointerAbbrev()); + writer.ref(ptr, Blob::writeInt); + } writer.emptyTag(); } }