mirror of
https://github.com/Eaglercraft-TeaVM-Fork/eagler-teavm.git
synced 2025-01-24 15:24:51 -08:00
Implementing simple mark&sweep GC
This commit is contained in:
parent
ae5d701aac
commit
3370898a54
|
@ -249,4 +249,22 @@ public final class WasmRuntime {
|
||||||
stack.add(-4).putInt(size);
|
stack.add(-4).putInt(size);
|
||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public static Address getStackGcRoots() {
|
||||||
|
return stack;
|
||||||
|
}
|
||||||
|
|
||||||
|
public static Address getNextStackRoots(Address address) {
|
||||||
|
int size = address.getInt() + 1;
|
||||||
|
return address.add(-size * 4);
|
||||||
|
}
|
||||||
|
|
||||||
|
public static int getStackRootCount(Address address) {
|
||||||
|
return address.getInt();
|
||||||
|
}
|
||||||
|
|
||||||
|
public static Address getStackRootPointer(Address address) {
|
||||||
|
int size = address.getInt();
|
||||||
|
return address.add(-size * 4);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -39,6 +39,7 @@ import org.teavm.backend.wasm.intrinsics.AddressIntrinsic;
|
||||||
import org.teavm.backend.wasm.intrinsics.AllocatorIntrinsic;
|
import org.teavm.backend.wasm.intrinsics.AllocatorIntrinsic;
|
||||||
import org.teavm.backend.wasm.intrinsics.ClassIntrinsic;
|
import org.teavm.backend.wasm.intrinsics.ClassIntrinsic;
|
||||||
import org.teavm.backend.wasm.intrinsics.FunctionIntrinsic;
|
import org.teavm.backend.wasm.intrinsics.FunctionIntrinsic;
|
||||||
|
import org.teavm.backend.wasm.intrinsics.GCIntrinsic;
|
||||||
import org.teavm.backend.wasm.intrinsics.PlatformClassIntrinsic;
|
import org.teavm.backend.wasm.intrinsics.PlatformClassIntrinsic;
|
||||||
import org.teavm.backend.wasm.intrinsics.PlatformIntrinsic;
|
import org.teavm.backend.wasm.intrinsics.PlatformIntrinsic;
|
||||||
import org.teavm.backend.wasm.intrinsics.PlatformObjectIntrinsic;
|
import org.teavm.backend.wasm.intrinsics.PlatformObjectIntrinsic;
|
||||||
|
@ -101,6 +102,7 @@ import org.teavm.model.instructions.CloneArrayInstruction;
|
||||||
import org.teavm.model.instructions.InitClassInstruction;
|
import org.teavm.model.instructions.InitClassInstruction;
|
||||||
import org.teavm.model.instructions.InvocationType;
|
import org.teavm.model.instructions.InvocationType;
|
||||||
import org.teavm.model.instructions.InvokeInstruction;
|
import org.teavm.model.instructions.InvokeInstruction;
|
||||||
|
import org.teavm.model.instructions.MutatorIntrinsic;
|
||||||
import org.teavm.model.lowlevel.ClassInitializerEliminator;
|
import org.teavm.model.lowlevel.ClassInitializerEliminator;
|
||||||
import org.teavm.model.lowlevel.ClassInitializerTransformer;
|
import org.teavm.model.lowlevel.ClassInitializerTransformer;
|
||||||
import org.teavm.model.lowlevel.GcRootMaintainingTransformer;
|
import org.teavm.model.lowlevel.GcRootMaintainingTransformer;
|
||||||
|
@ -201,6 +203,14 @@ public class WasmTarget implements TeaVMTarget {
|
||||||
Address.class, int.class, void.class), null).use();
|
Address.class, int.class, void.class), null).use();
|
||||||
dependencyChecker.linkMethod(new MethodReference(WasmRuntime.class, "allocStack",
|
dependencyChecker.linkMethod(new MethodReference(WasmRuntime.class, "allocStack",
|
||||||
int.class, Address.class), null).use();
|
int.class, Address.class), null).use();
|
||||||
|
dependencyChecker.linkMethod(new MethodReference(WasmRuntime.class, "getStackGcRoots", Address.class),
|
||||||
|
null) .use();
|
||||||
|
dependencyChecker.linkMethod(new MethodReference(WasmRuntime.class, "getNextStackRoots", Address.class,
|
||||||
|
Address.class), null).use();
|
||||||
|
dependencyChecker.linkMethod(new MethodReference(WasmRuntime.class, "getStackRootCount", Address.class,
|
||||||
|
int.class), null).use();
|
||||||
|
dependencyChecker.linkMethod(new MethodReference(WasmRuntime.class, "getStackRootPointer", Address.class,
|
||||||
|
Address.class), null).use();
|
||||||
|
|
||||||
dependencyChecker.linkMethod(new MethodReference(Allocator.class, "allocate",
|
dependencyChecker.linkMethod(new MethodReference(Allocator.class, "allocate",
|
||||||
RuntimeClass.class, Address.class), null).use();
|
RuntimeClass.class, Address.class), null).use();
|
||||||
|
@ -270,12 +280,17 @@ public class WasmTarget implements TeaVMTarget {
|
||||||
context.addIntrinsic(new AddressIntrinsic(classGenerator));
|
context.addIntrinsic(new AddressIntrinsic(classGenerator));
|
||||||
context.addIntrinsic(new StructureIntrinsic(classGenerator));
|
context.addIntrinsic(new StructureIntrinsic(classGenerator));
|
||||||
context.addIntrinsic(new FunctionIntrinsic(classGenerator));
|
context.addIntrinsic(new FunctionIntrinsic(classGenerator));
|
||||||
context.addIntrinsic(new WasmRuntimeIntrinsic());
|
WasmRuntimeIntrinsic wasmRuntimeIntrinsic = new WasmRuntimeIntrinsic();
|
||||||
|
context.addIntrinsic(wasmRuntimeIntrinsic);
|
||||||
context.addIntrinsic(new AllocatorIntrinsic(classGenerator));
|
context.addIntrinsic(new AllocatorIntrinsic(classGenerator));
|
||||||
context.addIntrinsic(new PlatformIntrinsic());
|
context.addIntrinsic(new PlatformIntrinsic());
|
||||||
context.addIntrinsic(new PlatformClassIntrinsic());
|
context.addIntrinsic(new PlatformClassIntrinsic());
|
||||||
context.addIntrinsic(new PlatformObjectIntrinsic(classGenerator));
|
context.addIntrinsic(new PlatformObjectIntrinsic(classGenerator));
|
||||||
context.addIntrinsic(new ClassIntrinsic());
|
context.addIntrinsic(new ClassIntrinsic());
|
||||||
|
GCIntrinsic gcIntrinsic = new GCIntrinsic();
|
||||||
|
context.addIntrinsic(gcIntrinsic);
|
||||||
|
MutatorIntrinsic mutatorIntrinsic = new MutatorIntrinsic();
|
||||||
|
context.addIntrinsic(mutatorIntrinsic);
|
||||||
|
|
||||||
WasmGenerator generator = new WasmGenerator(decompiler, classes,
|
WasmGenerator generator = new WasmGenerator(decompiler, classes,
|
||||||
context, classGenerator);
|
context, classGenerator);
|
||||||
|
@ -284,14 +299,14 @@ public class WasmTarget implements TeaVMTarget {
|
||||||
generateMethods(classes, context, generator, module);
|
generateMethods(classes, context, generator, module);
|
||||||
generateIsSupertypeFunctions(tagRegistry, module, classGenerator);
|
generateIsSupertypeFunctions(tagRegistry, module, classGenerator);
|
||||||
classGenerator.postProcess();
|
classGenerator.postProcess();
|
||||||
|
mutatorIntrinsic.setStaticGcRootsAddress(classGenerator.getStaticGcRootsAddress());
|
||||||
|
|
||||||
WasmMemorySegment dataSegment = new WasmMemorySegment();
|
WasmMemorySegment dataSegment = new WasmMemorySegment();
|
||||||
dataSegment.setData(binaryWriter.getData());
|
dataSegment.setData(binaryWriter.getData());
|
||||||
dataSegment.setOffset(256);
|
dataSegment.setOffset(256);
|
||||||
module.getSegments().add(dataSegment);
|
module.getSegments().add(dataSegment);
|
||||||
|
|
||||||
int address = renderStackInit(module, binaryWriter.getAddress());
|
renderMemoryLayout(module, binaryWriter.getAddress(), gcIntrinsic, wasmRuntimeIntrinsic);
|
||||||
renderAllocatorInit(module, address);
|
|
||||||
renderClinit(classes, classGenerator, module);
|
renderClinit(classes, classGenerator, module);
|
||||||
if (controller.wasCancelled()) {
|
if (controller.wasCancelled()) {
|
||||||
return;
|
return;
|
||||||
|
@ -370,14 +385,6 @@ public class WasmTarget implements TeaVMTarget {
|
||||||
for (String className : classes.getClassNames()) {
|
for (String className : classes.getClassNames()) {
|
||||||
ClassHolder cls = classes.get(className);
|
ClassHolder cls = classes.get(className);
|
||||||
for (MethodHolder method : cls.getMethods()) {
|
for (MethodHolder method : cls.getMethods()) {
|
||||||
if (method.getOwnerName().equals(Allocator.class.getName())
|
|
||||||
&& method.getName().equals("initialize")) {
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
if (method.getOwnerName().equals(WasmRuntime.class.getName())
|
|
||||||
&& method.getName().equals("initStack")) {
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
if (context.getIntrinsic(method.getReference()) != null) {
|
if (context.getIntrinsic(method.getReference()) != null) {
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
@ -599,24 +606,30 @@ public class WasmTarget implements TeaVMTarget {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private void renderAllocatorInit(WasmModule module, int address) {
|
private void renderMemoryLayout(WasmModule module, int address, GCIntrinsic gcIntrinsic,
|
||||||
WasmFunction function = new WasmFunction(WasmMangling.mangleMethod(new MethodReference(
|
WasmRuntimeIntrinsic runtimeIntrinsic) {
|
||||||
Allocator.class, "initialize", Address.class)));
|
|
||||||
function.setResult(WasmType.INT32);
|
|
||||||
function.getBody().add(new WasmReturn(new WasmInt32Constant(address)));
|
|
||||||
module.add(function);
|
|
||||||
}
|
|
||||||
|
|
||||||
private int renderStackInit(WasmModule module, int address) {
|
|
||||||
address = (((address - 1) / 256) + 1) * 256;
|
address = (((address - 1) / 256) + 1) * 256;
|
||||||
|
|
||||||
WasmFunction function = new WasmFunction(WasmMangling.mangleMethod(new MethodReference(
|
runtimeIntrinsic.setStackAddress(address);
|
||||||
WasmRuntime.class, "initStack", Address.class)));
|
address += 65536;
|
||||||
function.setResult(WasmType.INT32);
|
|
||||||
function.getBody().add(new WasmReturn(new WasmInt32Constant(address)));
|
|
||||||
module.add(function);
|
|
||||||
|
|
||||||
return address + 65536;
|
int gcMemory = module.getMemorySize() * 65536 - address;
|
||||||
|
int storageSize = (gcMemory >> 6) >> 2 << 2;
|
||||||
|
gcIntrinsic.setGCStorageAddress(address);
|
||||||
|
gcIntrinsic.setGCStorageSize(storageSize);
|
||||||
|
|
||||||
|
gcMemory -= storageSize;
|
||||||
|
address += storageSize;
|
||||||
|
int regionSize = 32768;
|
||||||
|
int regionCount = gcMemory / (2 + regionSize) + 1;
|
||||||
|
gcIntrinsic.setRegionSize(regionSize);
|
||||||
|
gcIntrinsic.setRegionsAddress(address);
|
||||||
|
gcIntrinsic.setRegionMaxCount(regionCount);
|
||||||
|
|
||||||
|
gcMemory -= regionCount * 2;
|
||||||
|
address += regionCount * 2;
|
||||||
|
gcIntrinsic.setHeapAddress(address);
|
||||||
|
gcIntrinsic.setAvailableBytes(address);
|
||||||
}
|
}
|
||||||
|
|
||||||
private VirtualTableProvider createVirtualTableProvider(ListableClassHolderSource classes) {
|
private VirtualTableProvider createVirtualTableProvider(ListableClassHolderSource classes) {
|
||||||
|
|
|
@ -113,6 +113,7 @@ import org.teavm.model.ValueType;
|
||||||
import org.teavm.model.classes.TagRegistry;
|
import org.teavm.model.classes.TagRegistry;
|
||||||
import org.teavm.model.classes.VirtualTableEntry;
|
import org.teavm.model.classes.VirtualTableEntry;
|
||||||
import org.teavm.runtime.Allocator;
|
import org.teavm.runtime.Allocator;
|
||||||
|
import org.teavm.runtime.Mutator;
|
||||||
import org.teavm.runtime.RuntimeArray;
|
import org.teavm.runtime.RuntimeArray;
|
||||||
import org.teavm.runtime.RuntimeClass;
|
import org.teavm.runtime.RuntimeClass;
|
||||||
|
|
||||||
|
@ -781,7 +782,7 @@ class WasmGenerationVisitor implements StatementVisitor, ExprVisitor {
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void visit(InvocationExpr expr) {
|
public void visit(InvocationExpr expr) {
|
||||||
if (expr.getMethod().getClassName().equals(Allocator.class.getName())) {
|
if (expr.getMethod().getClassName().equals(Mutator.class.getName())) {
|
||||||
switch (expr.getMethod().getName()) {
|
switch (expr.getMethod().getName()) {
|
||||||
case "allocStack":
|
case "allocStack":
|
||||||
generateAllocStack(expr.getArguments().get(0));
|
generateAllocStack(expr.getArguments().get(0));
|
||||||
|
@ -885,7 +886,7 @@ class WasmGenerationVisitor implements StatementVisitor, ExprVisitor {
|
||||||
|
|
||||||
private void generateAllocStack(Expr sizeExpr) {
|
private void generateAllocStack(Expr sizeExpr) {
|
||||||
if (stackVariable != null) {
|
if (stackVariable != null) {
|
||||||
throw new IllegalStateException("Call to Allocator.allocStack must be done only once");
|
throw new IllegalStateException("Call to Mutator.allocStack must be done only once");
|
||||||
}
|
}
|
||||||
stackVariable = getTemporary(WasmType.INT32);
|
stackVariable = getTemporary(WasmType.INT32);
|
||||||
stackVariable.setName("__stack__");
|
stackVariable.setName("__stack__");
|
||||||
|
@ -900,8 +901,8 @@ class WasmGenerationVisitor implements StatementVisitor, ExprVisitor {
|
||||||
|
|
||||||
private void generateReleaseStack() {
|
private void generateReleaseStack() {
|
||||||
if (stackVariable == null) {
|
if (stackVariable == null) {
|
||||||
throw new IllegalStateException("Call to Allocator.releaseStack must be dominated by "
|
throw new IllegalStateException("Call to Mutator.releaseStack must be dominated by "
|
||||||
+ "Allocator.allocStack");
|
+ "Mutator.allocStack");
|
||||||
}
|
}
|
||||||
|
|
||||||
int offset = classGenerator.getFieldOffset(new FieldReference(WasmRuntime.class.getName(), "stack"));
|
int offset = classGenerator.getFieldOffset(new FieldReference(WasmRuntime.class.getName(), "stack"));
|
||||||
|
@ -911,8 +912,8 @@ class WasmGenerationVisitor implements StatementVisitor, ExprVisitor {
|
||||||
|
|
||||||
private void generateRegisterGcRoot(Expr slotExpr, Expr gcRootExpr) {
|
private void generateRegisterGcRoot(Expr slotExpr, Expr gcRootExpr) {
|
||||||
if (stackVariable == null) {
|
if (stackVariable == null) {
|
||||||
throw new IllegalStateException("Call to Allocator.registerGcRoot must be dominated by "
|
throw new IllegalStateException("Call to Mutator.registerGcRoot must be dominated by "
|
||||||
+ "Allocator.allocStack");
|
+ "Mutator.allocStack");
|
||||||
}
|
}
|
||||||
|
|
||||||
slotExpr.acceptVisitor(this);
|
slotExpr.acceptVisitor(this);
|
||||||
|
@ -928,8 +929,8 @@ class WasmGenerationVisitor implements StatementVisitor, ExprVisitor {
|
||||||
|
|
||||||
private void generateRemoveGcRoot(Expr slotExpr) {
|
private void generateRemoveGcRoot(Expr slotExpr) {
|
||||||
if (stackVariable == null) {
|
if (stackVariable == null) {
|
||||||
throw new IllegalStateException("Call to Allocator.removeGcRoot must be dominated by "
|
throw new IllegalStateException("Call to Mutator.removeGcRoot must be dominated by "
|
||||||
+ "Allocator.allocStack");
|
+ "Mutator.allocStack");
|
||||||
}
|
}
|
||||||
|
|
||||||
slotExpr.acceptVisitor(this);
|
slotExpr.acceptVisitor(this);
|
||||||
|
|
|
@ -102,6 +102,7 @@ public class AddressIntrinsic implements WasmIntrinsic {
|
||||||
case "getChar":
|
case "getChar":
|
||||||
return new WasmLoadInt32(2, manager.generate(invocation.getArguments().get(0)),
|
return new WasmLoadInt32(2, manager.generate(invocation.getArguments().get(0)),
|
||||||
WasmInt32Subtype.UINT16);
|
WasmInt32Subtype.UINT16);
|
||||||
|
case "getAddress":
|
||||||
case "getInt":
|
case "getInt":
|
||||||
return new WasmLoadInt32(4, manager.generate(invocation.getArguments().get(0)),
|
return new WasmLoadInt32(4, manager.generate(invocation.getArguments().get(0)),
|
||||||
WasmInt32Subtype.INT32);
|
WasmInt32Subtype.INT32);
|
||||||
|
@ -122,6 +123,7 @@ public class AddressIntrinsic implements WasmIntrinsic {
|
||||||
WasmExpression value = manager.generate(invocation.getArguments().get(1));
|
WasmExpression value = manager.generate(invocation.getArguments().get(1));
|
||||||
return new WasmStoreInt32(2, address, value, WasmInt32Subtype.INT16);
|
return new WasmStoreInt32(2, address, value, WasmInt32Subtype.INT16);
|
||||||
}
|
}
|
||||||
|
case "putAddress":
|
||||||
case "putChar": {
|
case "putChar": {
|
||||||
WasmExpression address = manager.generate(invocation.getArguments().get(0));
|
WasmExpression address = manager.generate(invocation.getArguments().get(0));
|
||||||
WasmExpression value = manager.generate(invocation.getArguments().get(1));
|
WasmExpression value = manager.generate(invocation.getArguments().get(1));
|
||||||
|
@ -158,6 +160,11 @@ public class AddressIntrinsic implements WasmIntrinsic {
|
||||||
.collect(Collectors.toSet()));
|
.collect(Collectors.toSet()));
|
||||||
return call;
|
return call;
|
||||||
}
|
}
|
||||||
|
case "isLessThan": {
|
||||||
|
return new WasmIntBinary(WasmIntType.INT32, WasmIntBinaryOperation.LT_UNSIGNED,
|
||||||
|
manager.generate(invocation.getArguments().get(0)),
|
||||||
|
manager.generate(invocation.getArguments().get(1)));
|
||||||
|
}
|
||||||
default:
|
default:
|
||||||
throw new IllegalArgumentException(invocation.getMethod().toString());
|
throw new IllegalArgumentException(invocation.getMethod().toString());
|
||||||
}
|
}
|
||||||
|
|
|
@ -0,0 +1,132 @@
|
||||||
|
/*
|
||||||
|
* Copyright 2016 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.backend.wasm.intrinsics;
|
||||||
|
|
||||||
|
import java.util.ArrayList;
|
||||||
|
import java.util.List;
|
||||||
|
import org.teavm.ast.InvocationExpr;
|
||||||
|
import org.teavm.backend.wasm.model.expression.WasmExpression;
|
||||||
|
import org.teavm.backend.wasm.model.expression.WasmInt32Constant;
|
||||||
|
import org.teavm.backend.wasm.model.expression.WasmInt64Constant;
|
||||||
|
import org.teavm.model.MethodReference;
|
||||||
|
import org.teavm.runtime.GC;
|
||||||
|
|
||||||
|
public class GCIntrinsic implements WasmIntrinsic {
|
||||||
|
private List<WasmInt32Constant> heapAddressExpressions = new ArrayList<>();
|
||||||
|
private List<WasmInt64Constant> availableBytesExpressions = new ArrayList<>();
|
||||||
|
private List<WasmInt32Constant> gcStorageAddressExpressions = new ArrayList<>();
|
||||||
|
private List<WasmInt32Constant> gcStorageSizeExpressions = new ArrayList<>();
|
||||||
|
private List<WasmInt32Constant> regionSizeExpressions = new ArrayList<>();
|
||||||
|
private List<WasmInt32Constant> regionsAddressExpressions = new ArrayList<>();
|
||||||
|
private List<WasmInt32Constant> regionMaxCountExpressions = new ArrayList<>();
|
||||||
|
|
||||||
|
public void setHeapAddress(int address) {
|
||||||
|
for (WasmInt32Constant constant : heapAddressExpressions) {
|
||||||
|
constant.setValue(address);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setAvailableBytes(long availableBytes) {
|
||||||
|
for (WasmInt64Constant constant : availableBytesExpressions) {
|
||||||
|
constant.setValue(availableBytes);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setGCStorageAddress(int address) {
|
||||||
|
for (WasmInt32Constant constant : gcStorageAddressExpressions) {
|
||||||
|
constant.setValue(address);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setGCStorageSize(int storageSize) {
|
||||||
|
for (WasmInt32Constant constant : gcStorageSizeExpressions) {
|
||||||
|
constant.setValue(storageSize);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setRegionSize(int regionSize) {
|
||||||
|
for (WasmInt32Constant constant : regionSizeExpressions) {
|
||||||
|
constant.setValue(regionSize);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setRegionsAddress(int address) {
|
||||||
|
for (WasmInt32Constant constant : regionsAddressExpressions) {
|
||||||
|
constant.setValue(address);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setRegionMaxCount(int maxCount) {
|
||||||
|
for (WasmInt32Constant constant : regionMaxCountExpressions) {
|
||||||
|
constant.setValue(maxCount);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public boolean isApplicable(MethodReference methodReference) {
|
||||||
|
if (!methodReference.getClassName().endsWith(GC.class.getName())) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
switch (methodReference.getName()) {
|
||||||
|
case "gcStorageAddress":
|
||||||
|
case "gcStorageSize":
|
||||||
|
case "heapAddress":
|
||||||
|
case "availableBytes":
|
||||||
|
case "regionsAddress":
|
||||||
|
case "regionMaxCount":
|
||||||
|
case "regionSize":
|
||||||
|
return true;
|
||||||
|
default:
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public WasmExpression apply(InvocationExpr invocation, WasmIntrinsicManager manager) {
|
||||||
|
List<WasmInt32Constant> list;
|
||||||
|
switch (invocation.getMethod().getName()) {
|
||||||
|
case "gcStorageAddress":
|
||||||
|
list = gcStorageAddressExpressions;
|
||||||
|
break;
|
||||||
|
case "gcStorageSize":
|
||||||
|
list = gcStorageSizeExpressions;
|
||||||
|
break;
|
||||||
|
case "heapAddress":
|
||||||
|
list = heapAddressExpressions;
|
||||||
|
break;
|
||||||
|
case "regionsAddress":
|
||||||
|
list = regionsAddressExpressions;
|
||||||
|
break;
|
||||||
|
case "regionMaxCount":
|
||||||
|
list = regionMaxCountExpressions;
|
||||||
|
break;
|
||||||
|
case "regionSize":
|
||||||
|
list = regionSizeExpressions;
|
||||||
|
break;
|
||||||
|
case "availableBytes": {
|
||||||
|
WasmInt64Constant constant = new WasmInt64Constant(0);
|
||||||
|
availableBytesExpressions.add(constant);
|
||||||
|
return constant;
|
||||||
|
}
|
||||||
|
default:
|
||||||
|
throw new IllegalArgumentException(invocation.getMethod().toString());
|
||||||
|
}
|
||||||
|
WasmInt32Constant result = new WasmInt32Constant(0);
|
||||||
|
list.add(result);
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
}
|
|
@ -15,6 +15,8 @@
|
||||||
*/
|
*/
|
||||||
package org.teavm.backend.wasm.intrinsics;
|
package org.teavm.backend.wasm.intrinsics;
|
||||||
|
|
||||||
|
import java.util.ArrayList;
|
||||||
|
import java.util.List;
|
||||||
import org.teavm.ast.InvocationExpr;
|
import org.teavm.ast.InvocationExpr;
|
||||||
import org.teavm.backend.wasm.WasmRuntime;
|
import org.teavm.backend.wasm.WasmRuntime;
|
||||||
import org.teavm.backend.wasm.generate.WasmGeneratorUtil;
|
import org.teavm.backend.wasm.generate.WasmGeneratorUtil;
|
||||||
|
@ -23,40 +25,62 @@ import org.teavm.backend.wasm.model.expression.WasmExpression;
|
||||||
import org.teavm.backend.wasm.model.expression.WasmFloatBinary;
|
import org.teavm.backend.wasm.model.expression.WasmFloatBinary;
|
||||||
import org.teavm.backend.wasm.model.expression.WasmFloatBinaryOperation;
|
import org.teavm.backend.wasm.model.expression.WasmFloatBinaryOperation;
|
||||||
import org.teavm.backend.wasm.model.expression.WasmFloatType;
|
import org.teavm.backend.wasm.model.expression.WasmFloatType;
|
||||||
|
import org.teavm.backend.wasm.model.expression.WasmInt32Constant;
|
||||||
import org.teavm.backend.wasm.model.expression.WasmIntBinary;
|
import org.teavm.backend.wasm.model.expression.WasmIntBinary;
|
||||||
import org.teavm.backend.wasm.model.expression.WasmIntBinaryOperation;
|
import org.teavm.backend.wasm.model.expression.WasmIntBinaryOperation;
|
||||||
import org.teavm.backend.wasm.model.expression.WasmIntType;
|
import org.teavm.backend.wasm.model.expression.WasmIntType;
|
||||||
import org.teavm.model.MethodReference;
|
import org.teavm.model.MethodReference;
|
||||||
|
|
||||||
public class WasmRuntimeIntrinsic implements WasmIntrinsic {
|
public class WasmRuntimeIntrinsic implements WasmIntrinsic {
|
||||||
|
private List<WasmInt32Constant> stackExpressions = new ArrayList<>();
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public boolean isApplicable(MethodReference methodReference) {
|
public boolean isApplicable(MethodReference methodReference) {
|
||||||
return methodReference.getClassName().equals(WasmRuntime.class.getName())
|
if (!methodReference.getClassName().equals(WasmRuntime.class.getName())) {
|
||||||
&& (methodReference.getName().equals("lt") || methodReference.getName().equals("gt"));
|
return false;
|
||||||
|
}
|
||||||
|
switch (methodReference.getName()) {
|
||||||
|
case "gt":
|
||||||
|
case "lt":
|
||||||
|
case "initStack":
|
||||||
|
return true;
|
||||||
|
default:
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setStackAddress(int stackAddress) {
|
||||||
|
for (WasmInt32Constant constant : stackExpressions) {
|
||||||
|
constant.setValue(stackAddress);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public WasmExpression apply(InvocationExpr invocation, WasmIntrinsicManager manager) {
|
public WasmExpression apply(InvocationExpr invocation, WasmIntrinsicManager manager) {
|
||||||
|
switch (invocation.getMethod().getName()) {
|
||||||
|
case "lt":
|
||||||
|
return comparison(WasmIntBinaryOperation.LT_SIGNED, WasmFloatBinaryOperation.LT,
|
||||||
|
invocation, manager);
|
||||||
|
case "gt":
|
||||||
|
return comparison(WasmIntBinaryOperation.GT_SIGNED, WasmFloatBinaryOperation.GT,
|
||||||
|
invocation, manager);
|
||||||
|
case "initStack": {
|
||||||
|
WasmInt32Constant constant = new WasmInt32Constant(0);
|
||||||
|
stackExpressions.add(constant);
|
||||||
|
return constant;
|
||||||
|
}
|
||||||
|
default:
|
||||||
|
throw new IllegalArgumentException(invocation.getMethod().getName());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private static WasmExpression comparison(WasmIntBinaryOperation intOp, WasmFloatBinaryOperation floatOp,
|
||||||
|
InvocationExpr invocation, WasmIntrinsicManager manager) {
|
||||||
WasmType type = WasmGeneratorUtil.mapType(invocation.getMethod().parameterType(0));
|
WasmType type = WasmGeneratorUtil.mapType(invocation.getMethod().parameterType(0));
|
||||||
|
|
||||||
WasmExpression first = manager.generate(invocation.getArguments().get(0));
|
WasmExpression first = manager.generate(invocation.getArguments().get(0));
|
||||||
WasmExpression second = manager.generate(invocation.getArguments().get(1));
|
WasmExpression second = manager.generate(invocation.getArguments().get(1));
|
||||||
|
|
||||||
WasmIntBinaryOperation intOp;
|
|
||||||
WasmFloatBinaryOperation floatOp;
|
|
||||||
switch (invocation.getMethod().getName()) {
|
|
||||||
case "lt":
|
|
||||||
intOp = WasmIntBinaryOperation.LT_SIGNED;
|
|
||||||
floatOp = WasmFloatBinaryOperation.LT;
|
|
||||||
break;
|
|
||||||
case "gt":
|
|
||||||
intOp = WasmIntBinaryOperation.GT_SIGNED;
|
|
||||||
floatOp = WasmFloatBinaryOperation.GT;
|
|
||||||
break;
|
|
||||||
default:
|
|
||||||
throw new IllegalArgumentException(invocation.getMethod().getName());
|
|
||||||
}
|
|
||||||
|
|
||||||
switch (type) {
|
switch (type) {
|
||||||
case INT32:
|
case INT32:
|
||||||
return new WasmIntBinary(WasmIntType.INT32, intOp, first, second);
|
return new WasmIntBinary(WasmIntType.INT32, intOp, first, second);
|
||||||
|
@ -66,8 +90,8 @@ public class WasmRuntimeIntrinsic implements WasmIntrinsic {
|
||||||
return new WasmFloatBinary(WasmFloatType.FLOAT32, floatOp, first, second);
|
return new WasmFloatBinary(WasmFloatType.FLOAT32, floatOp, first, second);
|
||||||
case FLOAT64:
|
case FLOAT64:
|
||||||
return new WasmFloatBinary(WasmFloatType.FLOAT64, floatOp, first, second);
|
return new WasmFloatBinary(WasmFloatType.FLOAT64, floatOp, first, second);
|
||||||
}
|
default:
|
||||||
|
throw new IllegalArgumentException(type.toString());
|
||||||
return null;
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -0,0 +1,75 @@
|
||||||
|
/*
|
||||||
|
* Copyright 2016 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.model.instructions;
|
||||||
|
|
||||||
|
import java.util.ArrayList;
|
||||||
|
import java.util.List;
|
||||||
|
import org.teavm.ast.*;
|
||||||
|
import org.teavm.ast.InvocationType;
|
||||||
|
import org.teavm.backend.wasm.WasmRuntime;
|
||||||
|
import org.teavm.backend.wasm.intrinsics.WasmIntrinsic;
|
||||||
|
import org.teavm.backend.wasm.intrinsics.WasmIntrinsicManager;
|
||||||
|
import org.teavm.backend.wasm.model.expression.WasmExpression;
|
||||||
|
import org.teavm.backend.wasm.model.expression.WasmInt32Constant;
|
||||||
|
import org.teavm.model.MethodReference;
|
||||||
|
import org.teavm.runtime.Mutator;
|
||||||
|
|
||||||
|
public class MutatorIntrinsic implements WasmIntrinsic {
|
||||||
|
private List<WasmInt32Constant> staticGcRootsExpressions = new ArrayList<>();
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public boolean isApplicable(MethodReference methodReference) {
|
||||||
|
if (!methodReference.getClassName().equals(Mutator.class.getName())) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
switch (methodReference.getName()) {
|
||||||
|
case "getStaticGcRoots":
|
||||||
|
case "getStackGcRoots":
|
||||||
|
case "getNextStackRoots":
|
||||||
|
case "getStackRootCount":
|
||||||
|
case "getStackRootPointer":
|
||||||
|
return true;
|
||||||
|
default:
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setStaticGcRootsAddress(int address) {
|
||||||
|
for (WasmInt32Constant constant : staticGcRootsExpressions) {
|
||||||
|
constant.setValue(address);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public WasmExpression apply(InvocationExpr invocation, WasmIntrinsicManager manager) {
|
||||||
|
switch (invocation.getMethod().getName()) {
|
||||||
|
case "getStaticGcRoots": {
|
||||||
|
WasmInt32Constant constant = new WasmInt32Constant(0);
|
||||||
|
staticGcRootsExpressions.add(constant);
|
||||||
|
return constant;
|
||||||
|
}
|
||||||
|
default: {
|
||||||
|
InvocationExpr expr = new InvocationExpr();
|
||||||
|
MethodReference method = new MethodReference(WasmRuntime.class.getName(),
|
||||||
|
invocation.getMethod().getDescriptor());
|
||||||
|
expr.setMethod(method);
|
||||||
|
expr.setType(InvocationType.SPECIAL);
|
||||||
|
expr.getArguments().addAll(invocation.getArguments());
|
||||||
|
return manager.generate(expr);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
|
@ -48,7 +48,7 @@ import org.teavm.model.util.LivenessAnalyzer;
|
||||||
import org.teavm.model.util.ProgramUtils;
|
import org.teavm.model.util.ProgramUtils;
|
||||||
import org.teavm.model.util.TypeInferer;
|
import org.teavm.model.util.TypeInferer;
|
||||||
import org.teavm.model.util.VariableType;
|
import org.teavm.model.util.VariableType;
|
||||||
import org.teavm.runtime.Allocator;
|
import org.teavm.runtime.Mutator;
|
||||||
|
|
||||||
public class GcRootMaintainingTransformer {
|
public class GcRootMaintainingTransformer {
|
||||||
private ClassReaderSource classSource;
|
private ClassReaderSource classSource;
|
||||||
|
@ -158,7 +158,7 @@ public class GcRootMaintainingTransformer {
|
||||||
|
|
||||||
InvokeInstruction registerInvocation = new InvokeInstruction();
|
InvokeInstruction registerInvocation = new InvokeInstruction();
|
||||||
registerInvocation.setType(InvocationType.SPECIAL);
|
registerInvocation.setType(InvocationType.SPECIAL);
|
||||||
registerInvocation.setMethod(new MethodReference(Allocator.class, "registerGcRoot", int.class,
|
registerInvocation.setMethod(new MethodReference(Mutator.class, "registerGcRoot", int.class,
|
||||||
Object.class, void.class));
|
Object.class, void.class));
|
||||||
registerInvocation.getArguments().add(slotVar);
|
registerInvocation.getArguments().add(slotVar);
|
||||||
registerInvocation.getArguments().add(program.variableAt(liveVar));
|
registerInvocation.getArguments().add(program.variableAt(liveVar));
|
||||||
|
@ -177,7 +177,7 @@ public class GcRootMaintainingTransformer {
|
||||||
|
|
||||||
InvokeInstruction clearInvocation = new InvokeInstruction();
|
InvokeInstruction clearInvocation = new InvokeInstruction();
|
||||||
clearInvocation.setType(InvocationType.SPECIAL);
|
clearInvocation.setType(InvocationType.SPECIAL);
|
||||||
clearInvocation.setMethod(new MethodReference(Allocator.class, "removeGcRoot", int.class, void.class));
|
clearInvocation.setMethod(new MethodReference(Mutator.class, "removeGcRoot", int.class, void.class));
|
||||||
clearInvocation.getArguments().add(slotVar);
|
clearInvocation.getArguments().add(slotVar);
|
||||||
clearInvocation.setLocation(callInstruction.getLocation());
|
clearInvocation.setLocation(callInstruction.getLocation());
|
||||||
instructionsToAdd.add(clearInvocation);
|
instructionsToAdd.add(clearInvocation);
|
||||||
|
@ -200,7 +200,7 @@ public class GcRootMaintainingTransformer {
|
||||||
|
|
||||||
InvokeInstruction invocation = new InvokeInstruction();
|
InvokeInstruction invocation = new InvokeInstruction();
|
||||||
invocation.setType(InvocationType.SPECIAL);
|
invocation.setType(InvocationType.SPECIAL);
|
||||||
invocation.setMethod(new MethodReference(Allocator.class, "allocStack", int.class, void.class));
|
invocation.setMethod(new MethodReference(Mutator.class, "allocStack", int.class, void.class));
|
||||||
invocation.getArguments().add(sizeVariable);
|
invocation.getArguments().add(sizeVariable);
|
||||||
instructionsToAdd.add(invocation);
|
instructionsToAdd.add(invocation);
|
||||||
|
|
||||||
|
@ -261,7 +261,7 @@ public class GcRootMaintainingTransformer {
|
||||||
|
|
||||||
InvokeInstruction invocation = new InvokeInstruction();
|
InvokeInstruction invocation = new InvokeInstruction();
|
||||||
invocation.setType(InvocationType.SPECIAL);
|
invocation.setType(InvocationType.SPECIAL);
|
||||||
invocation.setMethod(new MethodReference(Allocator.class, "releaseStack", int.class, void.class));
|
invocation.setMethod(new MethodReference(Mutator.class, "releaseStack", int.class, void.class));
|
||||||
invocation.getArguments().add(sizeVariable);
|
invocation.getArguments().add(sizeVariable);
|
||||||
instructionsToAdd.add(invocation);
|
instructionsToAdd.add(invocation);
|
||||||
|
|
||||||
|
|
|
@ -31,7 +31,9 @@ public final class GC {
|
||||||
static FreeChunkHolder currentChunkPointer;
|
static FreeChunkHolder currentChunkPointer;
|
||||||
static int freeChunks;
|
static int freeChunks;
|
||||||
|
|
||||||
private static native Address gcStorageAddress();
|
static native Address gcStorageAddress();
|
||||||
|
|
||||||
|
static native int gcStorageSize();
|
||||||
|
|
||||||
private static native Address heapAddress();
|
private static native Address heapAddress();
|
||||||
|
|
||||||
|
@ -39,14 +41,14 @@ public final class GC {
|
||||||
|
|
||||||
private static native int regionMaxCount();
|
private static native int regionMaxCount();
|
||||||
|
|
||||||
private static native int availableBytes();
|
private static native long availableBytes();
|
||||||
|
|
||||||
private static native int regionSize();
|
private static native int regionSize();
|
||||||
|
|
||||||
static {
|
static {
|
||||||
currentChunk = heapAddress().toStructure();
|
currentChunk = heapAddress().toStructure();
|
||||||
currentChunk.classReference = 0;
|
currentChunk.classReference = 0;
|
||||||
currentChunk.size = availableBytes();
|
currentChunk.size = (int) availableBytes();
|
||||||
currentChunkLimit = currentChunk.toAddress().add(currentChunk.size);
|
currentChunkLimit = currentChunk.toAddress().add(currentChunk.size);
|
||||||
currentChunkPointer = gcStorageAddress().toStructure();
|
currentChunkPointer = gcStorageAddress().toStructure();
|
||||||
currentChunkPointer.value = currentChunk;
|
currentChunkPointer.value = currentChunk;
|
||||||
|
@ -90,7 +92,8 @@ public final class GC {
|
||||||
|
|
||||||
private static boolean collectGarbage(int size) {
|
private static boolean collectGarbage(int size) {
|
||||||
mark();
|
mark();
|
||||||
return false;
|
sweep();
|
||||||
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
private static void mark() {
|
private static void mark() {
|
||||||
|
@ -159,6 +162,150 @@ public final class GC {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private static void sweep() {
|
||||||
|
FreeChunkHolder freeChunkPtr = gcStorageAddress().toStructure();
|
||||||
|
freeChunks = 0;
|
||||||
|
|
||||||
|
RuntimeObject object = heapAddress().toStructure();
|
||||||
|
FreeChunk lastFreeSpace = null;
|
||||||
|
long heapSize = availableBytes();
|
||||||
|
long reclaimedSpace = 0;
|
||||||
|
long maxFreeChunk = 0;
|
||||||
|
int currentRegionIndex = 0;
|
||||||
|
int regionsCount = (int) ((heapSize - 1) / regionSize()) + 1;
|
||||||
|
Address currentRegionEnd = object.toAddress().add(regionSize());
|
||||||
|
Address limit = heapAddress().add(heapSize);
|
||||||
|
|
||||||
|
loop: while (object.toAddress().isLessThan(limit)) {
|
||||||
|
int tag = object.classReference;
|
||||||
|
boolean free;
|
||||||
|
if (tag == 0) {
|
||||||
|
free = true;
|
||||||
|
} else {
|
||||||
|
free = (tag & RuntimeObject.GC_MARKED) == 0;
|
||||||
|
if (!free) {
|
||||||
|
tag &= ~RuntimeObject.GC_MARKED;
|
||||||
|
}
|
||||||
|
object.classReference = tag;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (free) {
|
||||||
|
if (lastFreeSpace == null) {
|
||||||
|
lastFreeSpace = (FreeChunk) object;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!object.toAddress().isLessThan(currentRegionEnd)) {
|
||||||
|
currentRegionIndex = (int) ((object.toAddress().toLong() - heapAddress().toLong()) / regionSize());
|
||||||
|
Region currentRegion = regionsAddress().toAddress().add(Region.class, currentRegionIndex)
|
||||||
|
.toStructure();
|
||||||
|
if (currentRegion.start == 0) {
|
||||||
|
do {
|
||||||
|
if (++currentRegionIndex == regionsCount) {
|
||||||
|
object = limit.toStructure();
|
||||||
|
break loop;
|
||||||
|
}
|
||||||
|
currentRegion = regionsAddress().toAddress().add(Region.class, currentRegionIndex)
|
||||||
|
.toStructure();
|
||||||
|
} while (currentRegion.start == 0);
|
||||||
|
}
|
||||||
|
currentRegionEnd = currentRegion.toAddress().add(regionSize());
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
if (lastFreeSpace != null) {
|
||||||
|
lastFreeSpace.size = (int) (object.toAddress().toLong() - lastFreeSpace.toAddress().toLong());
|
||||||
|
freeChunkPtr.value = lastFreeSpace;
|
||||||
|
freeChunkPtr = freeChunkPtr.toAddress().add(FreeChunkHolder.class, 1).toStructure();
|
||||||
|
freeChunks++;
|
||||||
|
reclaimedSpace += lastFreeSpace.size;
|
||||||
|
if (maxFreeChunk < lastFreeSpace.size) {
|
||||||
|
maxFreeChunk = lastFreeSpace.size;
|
||||||
|
}
|
||||||
|
lastFreeSpace = null;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
int size = objectSize(object);
|
||||||
|
object = object.toAddress().add(size).toStructure();
|
||||||
|
}
|
||||||
|
|
||||||
|
if (lastFreeSpace != null) {
|
||||||
|
int freeSize = (int) (object.toAddress().toLong() - lastFreeSpace.toAddress().toLong());
|
||||||
|
lastFreeSpace.size = freeSize;
|
||||||
|
freeChunkPtr.value = lastFreeSpace;
|
||||||
|
freeChunkPtr = freeChunkPtr.toAddress().add(FreeChunkHolder.class, 1).toStructure();
|
||||||
|
freeChunks++;
|
||||||
|
reclaimedSpace += freeSize;
|
||||||
|
if (maxFreeChunk < freeSize) {
|
||||||
|
maxFreeChunk = freeSize;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
currentChunkPointer = heapAddress().toStructure();
|
||||||
|
sortFreeChunks(0, freeChunks - 1);
|
||||||
|
}
|
||||||
|
|
||||||
|
private static void sortFreeChunks(int lower, int upper) {
|
||||||
|
int start = lower;
|
||||||
|
int end = upper;
|
||||||
|
int mid = (lower + upper) / 2;
|
||||||
|
|
||||||
|
FreeChunk midChunk = getFreeChunk(mid).value;
|
||||||
|
outer: while (true) {
|
||||||
|
while (true) {
|
||||||
|
if (lower == upper) {
|
||||||
|
break outer;
|
||||||
|
}
|
||||||
|
if (getFreeChunk(lower).value.size <= midChunk.size) {
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
++lower;
|
||||||
|
}
|
||||||
|
while (true) {
|
||||||
|
if (lower == upper) {
|
||||||
|
break outer;
|
||||||
|
}
|
||||||
|
if (getFreeChunk(upper).value.size > midChunk.size) {
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
--upper;
|
||||||
|
}
|
||||||
|
FreeChunk tmp = getFreeChunk(lower).value;
|
||||||
|
getFreeChunk(lower).value = getFreeChunk(upper).value;
|
||||||
|
getFreeChunk(upper).value = tmp;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (lower - start > 0) {
|
||||||
|
sortFreeChunks(start, lower);
|
||||||
|
}
|
||||||
|
if (end - lower - 1 > 0) {
|
||||||
|
sortFreeChunks(lower + 1, end);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private static FreeChunkHolder getFreeChunk(int index) {
|
||||||
|
return currentChunkPointer.toAddress().add(FreeChunkHolder.class, index).toStructure();
|
||||||
|
}
|
||||||
|
|
||||||
|
private static int objectSize(RuntimeObject object) {
|
||||||
|
if (object.classReference == 0) {
|
||||||
|
return ((FreeChunk) object).size;
|
||||||
|
} else {
|
||||||
|
RuntimeClass cls = RuntimeClass.getClass(object);
|
||||||
|
if (cls.itemType == null) {
|
||||||
|
return cls.size;
|
||||||
|
} else {
|
||||||
|
int itemSize = (cls.itemType.flags & RuntimeClass.PRIMITIVE) == 0
|
||||||
|
? Address.sizeOf()
|
||||||
|
: cls.itemType.size;
|
||||||
|
RuntimeArray array = (RuntimeArray) object;
|
||||||
|
Address address = Address.fromInt(Structure.sizeOf(RuntimeArray.class));
|
||||||
|
address = Address.align(address, itemSize);
|
||||||
|
address = address.add(itemSize * array.size);
|
||||||
|
return address.toInt();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
private static boolean isMarked(RuntimeObject object) {
|
private static boolean isMarked(RuntimeObject object) {
|
||||||
return (object.classReference & RuntimeObject.GC_MARKED) != 0;
|
return (object.classReference & RuntimeObject.GC_MARKED) != 0;
|
||||||
}
|
}
|
||||||
|
|
|
@ -23,26 +23,24 @@ final class MarkQueue {
|
||||||
|
|
||||||
private static int head;
|
private static int head;
|
||||||
private static int tail;
|
private static int tail;
|
||||||
|
private static int limit;
|
||||||
|
|
||||||
static void init() {
|
static void init() {
|
||||||
head = 0;
|
head = 0;
|
||||||
tail = 0;
|
tail = 0;
|
||||||
|
limit = GC.gcStorageSize() / Address.sizeOf();
|
||||||
}
|
}
|
||||||
|
|
||||||
private static native Address getBase();
|
|
||||||
|
|
||||||
private static native int getSize();
|
|
||||||
|
|
||||||
static void enqueue(RuntimeObject object) {
|
static void enqueue(RuntimeObject object) {
|
||||||
getBase().add(Address.sizeOf() * tail).putAddress(object.toAddress());
|
GC.gcStorageAddress().add(Address.sizeOf() * tail).putAddress(object.toAddress());
|
||||||
if (++tail >= getSize()) {
|
if (++tail >= limit) {
|
||||||
tail = 0;
|
tail = 0;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
static RuntimeObject dequeue() {
|
static RuntimeObject dequeue() {
|
||||||
Address result = getBase().add(Address.sizeOf() * head).getAddress();
|
Address result = GC.gcStorageAddress().add(Address.sizeOf() * head).getAddress();
|
||||||
if (++head >= getSize()) {
|
if (++head >= limit) {
|
||||||
head = 0;
|
head = 0;
|
||||||
}
|
}
|
||||||
return result.toStructure();
|
return result.toStructure();
|
||||||
|
|
Loading…
Reference in New Issue
Block a user