Merge branch 'master' into compiler

Conflicts:

	src/compile.cpp
	src/compiler.cpp
	src/compiler.h
This commit is contained in:
Joel Dice 2008-04-13 13:48:20 -06:00
commit 525f733171
60 changed files with 2259 additions and 936 deletions

View File

@ -519,31 +519,6 @@ Java_java_lang_Math_pow(JNIEnv*, jclass, jdouble val, jdouble exp)
return pow(val, exp);
}
extern "C" JNIEXPORT void JNICALL
Java_java_lang_Math_natRandomInitialize(JNIEnv*, jclass, jlong val)
{
#ifdef WIN32
srand(val);
#else
srand48(val);
#endif
}
extern "C" JNIEXPORT jdouble JNICALL
Java_java_lang_Math_natRandom(JNIEnv*, jclass)
{
#ifdef WIN32
double r = static_cast<double>(rand()) / static_cast<double>(RAND_MAX);
if (r < 0 or r >= 1) {
return 0;
} else {
return r;
}
#else
return drand48();
#endif
}
extern "C" JNIEXPORT jdouble JNICALL
Java_java_lang_Math_floor(JNIEnv*, jclass, jdouble val)
{

View File

@ -575,6 +575,13 @@ Java_java_nio_channels_SocketSelector_natDoSocketSelect(JNIEnv *e, jclass,
timeval time = { interval / 1000, (interval % 1000) * 1000 };
int r = ::select(max + 1, &(s->read), &(s->write), &(s->except), &time);
if (r < 0) {
if (errno != EINTR) {
throwIOException(e);
return 0;
}
}
#ifdef WIN32
if (FD_ISSET(s->control.writer(), &(s->write)) or
FD_ISSET(s->control.writer(), &(s->except)))
@ -618,11 +625,6 @@ Java_java_nio_channels_SocketSelector_natDoSocketSelect(JNIEnv *e, jclass,
}
}
if (r < 0) {
if (errno != EINTR) {
throwIOException(e);
}
}
return r;
}

View File

@ -21,6 +21,10 @@ public class ByteArrayInputStream extends InputStream {
this.limit = offset + length;
}
public ByteArrayInputStream(byte[] array) {
this(array, 0, array.length);
}
public int read() {
if (position < limit) {
return array[position++] & 0xff;

View File

@ -52,5 +52,17 @@ public abstract class InputStream {
return 0;
}
public void mark(int limit) {
// ignore
}
public void reset() throws IOException {
// ignore
}
public boolean markSupported() {
return false;
}
public void close() throws IOException { }
}

View File

@ -0,0 +1,21 @@
/* Copyright (c) 2008, Avian Contributors
Permission to use, copy, modify, and/or distribute this software
for any purpose with or without fee is hereby granted, provided
that the above copyright notice and this permission notice appear
in all copies.
There is NO WARRANTY for this software. See license.txt for
details. */
package java.lang;
public class ArithmeticException extends RuntimeException {
public ArithmeticException(String message) {
super(message);
}
public ArithmeticException() {
this(null);
}
}

View File

@ -11,6 +11,9 @@
package java.lang;
public final class Character implements Comparable<Character> {
public static final int MIN_RADIX = 2;
public static final int MAX_RADIX = 36;
public static final Class TYPE = Class.forCanonicalName("C");
private final char value;
@ -67,6 +70,23 @@ public final class Character implements Comparable<Character> {
return c >= '0' && c <= '9';
}
public static int digit(char c, int radix) {
int digit = 0;
if ((c >= '0') && (c <= '9')) {
digit = c - '0';
} else if ((c >= 'a') && (c <= 'z')) {
digit = c - 'a' + 10;
} else {
return -1;
}
if (digit < radix) {
return digit;
} else {
return -1;
}
}
public static boolean isLetter(char c) {
return (c >= 'a' && c <= 'z') || (c >= 'A' && c <= 'Z');
}

View File

@ -376,6 +376,10 @@ public final class Class <T> {
return flags;
}
public boolean isInterface() {
return (flags & Modifier.INTERFACE) != 0;
}
public Class getSuperclass() {
return super_;
}

View File

@ -58,6 +58,10 @@ public final class Integer extends Number implements Comparable<Integer> {
return Long.toString(((long) v) & 0xFFFFFFFFL, 16);
}
public static String toBinaryString(int v) {
return Long.toString(((long) v) & 0xFFFFFFFFL, 2);
}
public byte byteValue() {
return (byte) value;
}

View File

@ -11,8 +11,10 @@
package java.lang;
public final class Long extends Number implements Comparable<Long> {
public static final Long MIN_VALUE = -9223372036854775808l;
public static final Long MAX_VALUE = 9223372036854775807l;
public static final Class TYPE = Class.forCanonicalName("J");
public static final Long MAX_VALUE = 9223372036854775807l;
private final long value;
@ -131,16 +133,13 @@ public final class Long extends Number implements Comparable<Long> {
for (; i < s.length(); ++i) {
char c = s.charAt(i);
if (((c >= '0') && (c <= '9')) ||
((c >= 'a') && (c <= 'z'))) {
long digit = ((c >= '0' && c <= '9') ? (c - '0') : (c - 'a' + 10));
if (digit < radix) {
number += digit * pow(radix, (s.length() - i - 1));
continue;
}
int digit = Character.digit(c, radix);
if (digit >= 0) {
number += digit * pow(radix, (s.length() - i - 1));
} else {
throw new NumberFormatException("invalid character " + c + " code " +
(int) c);
}
throw new NumberFormatException("invalid character " + c + " code " +
(int) c);
}
if (negative) {

View File

@ -10,10 +10,12 @@
package java.lang;
import java.util.Random;
public final class Math {
public static final double E = 2.718281828459045;
public static final double PI = 3.141592653589793;
private static boolean randomInitialized = false;
private static final Random random = new Random();
private Math() { }
@ -74,17 +76,9 @@ public final class Math {
}
public static double random() {
if (randomInitialized) {
natRandomInitialize(System.currentTimeMillis());
randomInitialized = true;
}
return natRandom();
return random.nextDouble();
}
public static native void natRandomInitialize(long val);
public static native double natRandom();
public static native double floor(double v);
public static native double ceil(double v);

View File

@ -108,4 +108,14 @@ public class Thread implements Runnable {
t.sleepLock.wait(milliseconds);
}
}
public StackTraceElement[] getStackTrace() {
return Throwable.resolveTrace(getStackTrace(peer));
}
private static native Object getStackTrace(long peer);
public static native int activeCount();
public static native int enumerate(Thread[] array);
}

View File

@ -65,7 +65,7 @@ public class Throwable {
private static native Object trace(int skipCount);
private static native StackTraceElement[] resolveTrace(Object trace);
static native StackTraceElement[] resolveTrace(Object trace);
private StackTraceElement[] resolveTrace() {
if (! (trace instanceof StackTraceElement[])) {

View File

@ -33,4 +33,6 @@ public final class Modifier {
public static boolean isStatic (int v) { return (v & STATIC) != 0; }
public static boolean isFinal (int v) { return (v & FINAL) != 0; }
public static boolean isSuper (int v) { return (v & SUPER) != 0; }
public static boolean isNative (int v) { return (v & NATIVE) != 0; }
public static boolean isAbstract (int v) { return (v & ABSTRACT) != 0; }
}

View File

@ -10,7 +10,7 @@
package java.net;
public class InetSocketAddress {
public class InetSocketAddress extends SocketAddress {
private final String host;
private final int port;

View File

@ -0,0 +1,17 @@
/* Copyright (c) 2008, Avian Contributors
Permission to use, copy, modify, and/or distribute this software
for any purpose with or without fee is hereby granted, provided
that the above copyright notice and this permission notice appear
in all copies.
There is NO WARRANTY for this software. See license.txt for
details. */
package java.net;
import java.io.IOException;
public abstract class ServerSocket {
public abstract void bind(SocketAddress address) throws IOException;
}

View File

@ -0,0 +1,13 @@
/* Copyright (c) 2008, Avian Contributors
Permission to use, copy, modify, and/or distribute this software
for any purpose with or without fee is hereby granted, provided
that the above copyright notice and this permission notice appear
in all copies.
There is NO WARRANTY for this software. See license.txt for
details. */
package java.net;
public abstract class SocketAddress { }

View File

@ -0,0 +1,59 @@
/* Copyright (c) 2008, Avian Contributors
Permission to use, copy, modify, and/or distribute this software
for any purpose with or without fee is hereby granted, provided
that the above copyright notice and this permission notice appear
in all copies.
There is NO WARRANTY for this software. See license.txt for
details. */
package java.nio;
public abstract class Buffer {
protected int capacity;
protected int position;
protected int limit;
public int limit() {
return limit;
}
public int remaining() {
return limit-position;
}
public int position() {
return position;
}
public int capacity() {
return capacity;
}
public Buffer limit(int newLimit) {
limit = newLimit;
return this;
}
public Buffer position(int newPosition) {
position = newPosition;
return this;
}
public boolean hasRemaining() {
return remaining() > 0;
}
public Buffer clear() {
position = 0;
limit = capacity;
return this;
}
public Buffer flip() {
limit = position;
position = 0;
return this;
}
}

View File

@ -10,27 +10,28 @@
package java.nio;
public class ByteBuffer implements Comparable<ByteBuffer> {
public class ByteBuffer extends Buffer implements Comparable<ByteBuffer> {
private final byte[] array;
private int arrayOffset;
private int capacity;
private int position;
private int limit;
private final boolean readOnly;
public static ByteBuffer allocate(int capacity) {
return new ByteBuffer(new byte[capacity], false);
return new ByteBuffer(new byte[capacity], 0, capacity, false);
}
public static ByteBuffer wrap(byte[] array) {
return new ByteBuffer(array, false);
return wrap(array, 0, array.length);
}
private ByteBuffer(byte[] array, boolean readOnly) {
public static ByteBuffer wrap(byte[] array, int offset, int length) {
return new ByteBuffer(array, offset, length, false);
}
private ByteBuffer(byte[] array, int offset, int length, boolean readOnly) {
this.array = array;
this.readOnly = readOnly;
arrayOffset = 0;
capacity = array.length;
arrayOffset = offset;
capacity = length;
limit = capacity;
position = 0;
}
@ -55,35 +56,8 @@ public class ByteBuffer implements Comparable<ByteBuffer> {
return array;
}
public ByteBuffer clear() {
position = 0;
limit = capacity;
return this;
}
public ByteBuffer slice() {
ByteBuffer buf = new ByteBuffer(array, true);
buf.arrayOffset = arrayOffset + position;
buf.position = 0;
buf.capacity = remaining();
buf.limit = buf.capacity;
return buf;
}
public int limit() {
return limit;
}
public int remaining() {
return limit-position;
}
public int position() {
return position;
}
public int capacity() {
return capacity;
return new ByteBuffer(array, arrayOffset + position, remaining(), true);
}
public int arrayOffset() {
@ -100,16 +74,6 @@ public class ByteBuffer implements Comparable<ByteBuffer> {
return this;
}
public ByteBuffer limit(int newLimit) {
limit = newLimit;
return this;
}
public ByteBuffer position(int newPosition) {
position = newPosition;
return this;
}
public ByteBuffer put(byte val) {
checkPut(1);
array[arrayOffset+(position++)] = val;
@ -164,10 +128,6 @@ public class ByteBuffer implements Comparable<ByteBuffer> {
return this;
}
public boolean hasRemaining() {
return remaining() > 0;
}
public byte get() {
checkGet(1);
return array[arrayOffset+(position++)];
@ -189,12 +149,6 @@ public class ByteBuffer implements Comparable<ByteBuffer> {
return array[arrayOffset+position];
}
public ByteBuffer flip() {
limit = position;
position = 0;
return this;
}
public int getInt() {
checkGet(4);
int i = get() << 24;

View File

@ -16,22 +16,23 @@ import java.nio.ByteBuffer;
public abstract class SelectableChannel implements Channel {
private SelectionKey key;
public abstract int read(ByteBuffer b) throws Exception;
public abstract int write(ByteBuffer b) throws Exception;
public abstract boolean isOpen();
public abstract SelectableChannel configureBlocking(boolean v)
throws IOException;
public SelectionKey register(Selector selector, int interestOps,
Object attachment)
{
SelectionKey key = new SelectionKey
(this, selector, interestOps, attachment);
key = new SelectionKey(this, selector, interestOps, attachment);
selector.add(key);
return key;
}
public boolean isOpen() {
return key != null;
}
public void close() throws IOException {
if (key != null) {
key.selector().remove(key);
key = null;
}
}

View File

@ -36,8 +36,9 @@ public class SelectionKey {
return interestOps;
}
public void interestOps(int v) {
public SelectionKey interestOps(int v) {
this.interestOps = v;
return this;
}
public int readyOps() {

View File

@ -40,9 +40,9 @@ public abstract class Selector {
public abstract boolean isOpen();
public abstract void wakeup();
public abstract Selector wakeup();
public abstract void select(long interval) throws IOException;
public abstract int select(long interval) throws IOException;
public abstract void close();
}

View File

@ -10,7 +10,11 @@
package java.nio.channels;
import java.io.IOException;
import java.net.InetSocketAddress;
import java.net.SocketAddress;
import java.net.ServerSocket;
public class ServerSocketChannel extends SocketChannel {
public static ServerSocketChannel open() {
@ -24,26 +28,32 @@ public class ServerSocketChannel extends SocketChannel {
return c;
}
public Handle socket() {
public ServerSocket socket() {
return new Handle();
}
private int doAccept() throws Exception {
private int doAccept() throws IOException {
return natDoAccept(socket);
}
private int doListen(String host, int port) throws Exception {
private int doListen(String host, int port) throws IOException {
return natDoListen(host, port);
}
public class Handle {
public void bind(InetSocketAddress address)
throws Exception
public class Handle extends ServerSocket {
public void bind(SocketAddress address)
throws IOException
{
socket = doListen(address.getHostName(), address.getPort());
InetSocketAddress a;
try {
a = (InetSocketAddress) address;
} catch (ClassCastException e) {
throw new IllegalArgumentException();
}
socket = doListen(a.getHostName(), a.getPort());
}
}
private static native int natDoAccept(int socket) throws Exception;
private static native int natDoListen(String host, int port) throws Exception;
private static native int natDoAccept(int socket) throws IOException;
private static native int natDoListen(String host, int port) throws IOException;
}

View File

@ -11,6 +11,7 @@
package java.nio.channels;
import java.io.IOException;
import java.net.SocketAddress;
import java.net.InetSocketAddress;
import java.nio.ByteBuffer;
@ -20,30 +21,33 @@ public class SocketChannel extends SelectableChannel
public static final int InvalidSocket = -1;
protected int socket = InvalidSocket;
protected boolean open = true;
protected boolean connected = false;
public static SocketChannel open() {
return new SocketChannel();
}
public void configureBlocking(boolean v) {
public SelectableChannel configureBlocking(boolean v) {
if (v) throw new IllegalArgumentException();
return this;
}
public void connect(InetSocketAddress address) throws Exception {
socket = doConnect(address.getHostName(), address.getPort());
public boolean connect(SocketAddress address) throws Exception {
InetSocketAddress a;
try {
a = (InetSocketAddress) address;
} catch (ClassCastException e) {
throw new UnsupportedAddressTypeException();
}
socket = doConnect(a.getHostName(), a.getPort());
return connected;
}
public void close() throws IOException {
super.close();
if (! open) return;
closeSocket();
open = false;
}
public boolean isOpen() {
return open;
if (isOpen()) {
super.close();
closeSocket();
}
}
private int doConnect(String host, int port) throws Exception {
@ -54,7 +58,7 @@ public class SocketChannel extends SelectableChannel
}
public int read(ByteBuffer b) throws IOException {
if (! open) return -1;
if (! isOpen()) return -1;
if (b.remaining() == 0) return 0;
int r = natRead(socket, b.array(), b.arrayOffset() + b.position(), b.remaining());
if (r > 0) {

View File

@ -14,15 +14,10 @@ import java.io.IOException;
import java.util.Iterator;
class SocketSelector extends Selector {
private static final boolean IsWin32;
protected long state;
protected final Object lock = new Object();
protected boolean woken = false;
static {
IsWin32 = false;
}
public SocketSelector() {
state = natInit();
}
@ -31,7 +26,7 @@ class SocketSelector extends Selector {
return state != 0;
}
public void wakeup() {
public Selector wakeup() {
synchronized (lock) {
if (! woken) {
woken = true;
@ -39,6 +34,7 @@ class SocketSelector extends Selector {
natWakeup(state);
}
}
return this;
}
private boolean clearWoken() {
@ -52,10 +48,11 @@ class SocketSelector extends Selector {
}
}
public synchronized void select(long interval) throws IOException {
public synchronized int select(long interval) throws IOException {
selectedKeys.clear();
if (clearWoken()) return;
if (clearWoken()) return 0;
int max=0;
for (Iterator<SelectionKey> it = keys.iterator();
it.hasNext();)
@ -63,14 +60,14 @@ class SocketSelector extends Selector {
SelectionKey key = it.next();
SocketChannel c = (SocketChannel)key.channel();
int socket = c.socketFD();
if (! c.isOpen()) {
if (c.isOpen()) {
key.readyOps(0);
max = natSelectUpdateInterestSet
(socket, key.interestOps(), state, max);
} else {
natSelectClearAll(socket, state);
it.remove();
continue;
}
key.readyOps(0);
max = natSelectUpdateInterestSet(socket, key.interestOps(), state, max);
}
int r = natDoSocketSelect(state, max, interval);
@ -87,6 +84,8 @@ class SocketSelector extends Selector {
}
}
clearWoken();
return selectedKeys.size();
}
public void close() {

View File

@ -0,0 +1,17 @@
/* Copyright (c) 2008, Avian Contributors
Permission to use, copy, modify, and/or distribute this software
for any purpose with or without fee is hereby granted, provided
that the above copyright notice and this permission notice appear
in all copies.
There is NO WARRANTY for this software. See license.txt for
details. */
package java.nio.channels;
public class UnsupportedAddressTypeException extends IllegalArgumentException {
public UnsupportedAddressTypeException() {
super(null, null);
}
}

View File

@ -11,15 +11,54 @@
package java.util;
public class Random {
public int nextInt(int n) {
return nextInt() % n;
private static final long Mask = 0x5DEECE66DL;
private static long nextSeed = 0;
private long seed;
public Random(long seed) {
setSeed(seed);
}
public Random() {
setSeed((nextSeed++) ^ System.currentTimeMillis());
}
public void setSeed(long seed) {
this.seed = (seed ^ Mask) & ((1L << 48) - 1);
}
protected int next(int bits) {
seed = ((seed * Mask) + 0xBL) & ((1L << 48) - 1);
return (int) (seed >>> (48 - bits));
}
public int nextInt(int limit) {
if (limit <= 0) {
throw new IllegalArgumentException();
}
if ((limit & -limit) == limit) {
// limit is a power of two
return (int) ((limit * (long) next(31)) >> 31);
}
int bits;
int value;
do {
bits = next(31);
value = bits % limit;
} while (bits - value + (limit - 1) < 0);
return value;
}
public int nextInt() {
return (int)(Math.random()*Integer.MAX_VALUE);
return next(32);
}
public double nextDouble() {
return Math.random();
return (((long) next(26) << 27) + next(27)) / (double) (1L << 53);
}
}

View File

@ -37,6 +37,10 @@ public class Vector<T> implements List<T> {
list.add(index, element);
}
public void insertElementAt(T element, int index) {
add(index, element);
}
public synchronized boolean add(T element) {
return list.add(element);
}
@ -57,6 +61,10 @@ public class Vector<T> implements List<T> {
return list.set(index, value);
}
public T setElementAt(T value, int index) {
return set(index, value);
}
public T elementAt(int index) {
return get(index);
}

View File

@ -52,12 +52,12 @@ strip-all = --strip-all
rdynamic = -rdynamic
warnings = -Wall -Wextra -Werror -Wunused-parameter \
-Winit-self -Wconversion
warnings = -Wall -Wextra -Werror -Wunused-parameter -Winit-self
common-cflags = $(warnings) -fno-rtti -fno-exceptions \
-I$(JAVA_HOME)/include -idirafter $(src) -I$(native-build) \
-D__STDC_LIMIT_MACROS -D_JNI_IMPLEMENTATION_ -DAVIAN_VERSION=\"$(version)\"
-D__STDC_LIMIT_MACROS -D_JNI_IMPLEMENTATION_ -DAVIAN_VERSION=\"$(version)\" \
-DBOOT_CLASSPATH=\"[classpathJar]\"
build-cflags = $(common-cflags) -fPIC -fvisibility=hidden \
-I$(JAVA_HOME)/include/linux -I$(src) -pthread
@ -66,7 +66,7 @@ cflags = $(build-cflags)
common-lflags = -lm -lz
lflags = $(common-lflags) -lpthread -ldl -rdynamic
lflags = $(common-lflags) -lpthread -ldl
system = posix
asm = x86
@ -75,6 +75,11 @@ object-arch = i386:x86-64
object-format = elf64-x86-64
pointer-size = 8
so-prefix = lib
so-suffix = .so
shared = -shared
ifeq ($(arch),i386)
object-arch = i386
object-format = elf32-i386
@ -85,8 +90,11 @@ ifeq ($(platform),darwin)
build-cflags = $(common-cflags) -fPIC -fvisibility=hidden \
-I$(JAVA_HOME)/include/linux -I$(src)
lflags = $(common-lflags) -ldl -framework CoreFoundation
rdynamic =
strip-all = -S -x
binaryToMacho = $(native-build)/binaryToMacho
so-suffix = .jnilib
shared = -dynamiclib
endif
ifeq ($(platform),windows)
@ -96,6 +104,9 @@ ifeq ($(platform),windows)
system = windows
object-format = pe-i386
so-prefix =
so-suffix = .dll
cxx = i586-mingw32msvc-g++
cc = i586-mingw32msvc-gcc
dlltool = i586-mingw32msvc-dlltool
@ -104,7 +115,7 @@ ifeq ($(platform),windows)
objcopy = i586-mingw32msvc-objcopy
rdynamic = -Wl,--export-dynamic
lflags = -L$(lib) $(common-lflags) -lws2_32 -Wl,--kill-at -mwindows -mconsole
lflags = -L$(lib) $(common-lflags) -lws2_32 -mwindows -mconsole
cflags = $(common-cflags) -I$(inc)
endif
@ -185,9 +196,12 @@ vm-cpp-objects = $(call cpp-objects,$(vm-sources),$(src),$(native-build))
vm-asm-objects = $(call asm-objects,$(vm-asm-sources),$(src),$(native-build))
vm-objects = $(vm-cpp-objects) $(vm-asm-objects)
driver-sources = $(src)/main.cpp
driver-source = $(src)/main.cpp
driver-object = $(native-build)/main.o
driver-dynamic-object = $(native-build)/main-dynamic.o
driver-object = $(call cpp-objects,$(driver-sources),$(src),$(native-build))
boot-source = $(src)/boot.cpp
boot-object = $(native-build)/boot.o
generator-headers = $(src)/constants.h
generator-sources = $(src)/type-generator.cpp
@ -195,8 +209,10 @@ generator-objects = \
$(call cpp-objects,$(generator-sources),$(src),$(native-build))
generator = $(native-build)/generator
libvm = $(native-build)/lib$(name).a
vm = $(native-build)/$(name)
static-library = $(native-build)/lib$(name).a
executable = $(native-build)/$(name)
dynamic-library = $(native-build)/$(so-prefix)$(name)$(so-suffix)
executable-dynamic = $(native-build)/$(name)-dynamic
classpath-sources = $(shell find $(classpath) -name '*.java')
classpath-classes = \
@ -216,31 +232,36 @@ flags = -cp $(test-build)
args = $(flags) $(input)
.PHONY: build
build: $(vm) $(libvm) $(classpath-dep) $(test-dep)
build: $(static-library) $(executable) $(dynamic-library) \
$(executable-dynamic) $(classpath-dep) $(test-dep)
$(test-classes): $(classpath-dep)
.PHONY: run
run: build
$(vm) $(args)
$(executable) $(args)
.PHONY: debug
debug: build
gdb --args $(vm) $(args)
gdb --args $(executable) $(args)
.PHONY: vg
vg: build
$(vg) $(vm) $(args)
$(vg) $(executable) $(args)
.PHONY: test
test: build
/bin/bash $(test)/test.sh 2>/dev/null \
$(vm) $(mode) "$(flags)" \
$(executable) $(mode) "$(flags)" \
$(call class-names,$(test-build),$(test-classes))
.PHONY: javadoc
javadoc:
javadoc -sourcepath classpath -d build/javadoc -subpackages java
javadoc -sourcepath classpath -d build/javadoc -subpackages java \
-windowtitle "Avian v$(version) Class Library API" \
-doctitle "Avian v$(version) Class Library API" \
-header "Avian v$(version)" \
-bottom "<a href=\"http://oss.readytalk.com/avian/\">http://oss.readytalk.com/avian</a>"
.PHONY: clean
clean:
@ -293,7 +314,16 @@ $(vm-cpp-objects): $(native-build)/%.o: $(src)/%.cpp $(vm-depends)
$(vm-asm-objects): $(native-build)/%-asm.o: $(src)/%.S
$(compile-object)
$(driver-object): $(native-build)/%.o: $(src)/%.cpp
$(driver-object): $(driver-source)
$(compile-object)
$(driver-dynamic-object): $(driver-source)
@echo "compiling $(@)"
@mkdir -p $(dir $(@))
$(cxx) $(cflags) -DBOOT_LIBRARY=\"$(so-prefix)$(name)$(so-suffix)\" \
-c $(<) -o $(@)
$(boot-object): $(boot-source)
$(compile-object)
$(build)/classpath.jar: $(classpath-dep)
@ -305,9 +335,10 @@ $(binaryToMacho): $(src)/binaryToMacho.cpp
$(cxx) $(^) -o $(@)
$(classpath-object): $(build)/classpath.jar $(binaryToMacho)
@echo "creating $(@)"
ifeq ($(platform),darwin)
$(binaryToMacho) $(build)/classpath.jar \
__binary_classpath_jar_start __binary_classpath_jar_size > $(@)
__binary_classpath_jar_start __binary_classpath_jar_end > $(@)
else
(wd=$$(pwd); \
cd $(build); \
@ -324,23 +355,37 @@ $(generator-objects): $(native-build)/%.o: $(src)/%.cpp
$(jni-objects): $(native-build)/%.o: $(classpath)/%.cpp
$(compile-object)
$(libvm): $(vm-objects) $(jni-objects)
$(static-library): $(vm-objects) $(jni-objects)
@echo "creating $(@)"
rm -rf $(@)
$(ar) cru $(@) $(^)
$(ranlib) $(@)
$(vm): $(vm-objects) $(classpath-object) $(jni-objects) $(driver-object)
$(executable): \
$(vm-objects) $(classpath-object) $(jni-objects) $(driver-object) \
$(boot-object)
@echo "linking $(@)"
ifeq ($(platform),windows)
$(dlltool) -z $(@).def $(^)
$(dlltool) -k -d $(@).def -e $(@).exp
$(cc) $(@).exp $(^) $(lflags) -o $(@)
else
$(cc) $(^) $(lflags) -o $(@)
$(cc) $(^) $(rdynamic) $(lflags) -o $(@)
endif
$(strip) $(strip-all) $(@)
$(dynamic-library): \
$(vm-objects) $(classpath-object) $(dynamic-object) $(jni-objects) \
$(boot-object)
@echo "linking $(@)"
$(cc) $(^) $(shared) $(lflags) -o $(@)
$(strip) $(strip-all) $(@)
$(executable-dynamic): $(driver-dynamic-object) $(dynamic-library)
@echo "linking $(@)"
$(cc) $(^) $(lflags) -o $(@)
$(strip) $(strip-all) $(@)
$(generator): $(generator-objects)
@echo "linking $(@)"
$(build-cc) $(^) -o $(@)

View File

@ -165,12 +165,12 @@ setting the classpath to "[bootJar]".
extern "C" {
extern const uint8_t SYMBOL(start)[];
extern const uint8_t SYMBOL(size)[];
extern const uint8_t SYMBOL(end)[];
EXPORT const uint8_t*
bootJar(unsigned* size)
{
*size = reinterpret_cast<uintptr_t>(SYMBOL(size));
*size = SYMBOL(end) - SYMBOL(start);
return SYMBOL(start);
}

View File

@ -18,9 +18,9 @@ namespace vm {
class Allocator {
public:
virtual ~Allocator() { }
virtual void* tryAllocate(unsigned size, bool executable) = 0;
virtual void* allocate(unsigned size, bool executable) = 0;
virtual void free(const void* p, unsigned size, bool executable) = 0;
virtual void* tryAllocate(unsigned size) = 0;
virtual void* allocate(unsigned size) = 0;
virtual void free(const void* p, unsigned size) = 0;
};
} // namespace vm

View File

@ -120,17 +120,14 @@ class Assembler {
class Memory: public Operand {
public:
Memory(int base, int offset, int index = NoRegister, unsigned scale = 0,
TraceHandler* traceHandler = 0):
base(base), offset(offset), index(index), scale(scale),
traceHandler(traceHandler)
Memory(int base, int offset, int index = NoRegister, unsigned scale = 0):
base(base), offset(offset), index(index), scale(scale)
{ }
int base;
int offset;
int index;
unsigned scale;
TraceHandler* traceHandler;
};
class Client {

View File

@ -30,10 +30,10 @@ pad(unsigned n)
void
writeObject(FILE* out, const uint8_t* data, unsigned size,
const char* dataName, const char* sizeName)
const char* startName, const char* endName)
{
unsigned dataNameLength = strlen(dataName) + 1;
unsigned sizeNameLength = strlen(sizeName) + 1;
unsigned startNameLength = strlen(startName) + 1;
unsigned endNameLength = strlen(endName) + 1;
mach_header header = {
MH_MAGIC, // magic
@ -96,7 +96,7 @@ writeObject(FILE* out, const uint8_t* data, unsigned size,
+ sizeof(symtab_command)
+ pad(size)
+ (sizeof(struct nlist) * 2), // stroff
1 + dataNameLength + sizeNameLength, // strsize
1 + startNameLength + endNameLength, // strsize
};
struct nlist symbolList[] = {
@ -108,9 +108,9 @@ writeObject(FILE* out, const uint8_t* data, unsigned size,
0 // n_value
},
{
reinterpret_cast<char*>(1 + dataNameLength), // n_un
N_ABS | N_EXT, // n_type
NO_SECT, // n_sect
reinterpret_cast<char*>(1 + startNameLength), // n_un
N_SECT | N_EXT, // n_type
1, // n_sect
0, // n_desc
size // n_value
}
@ -127,8 +127,8 @@ writeObject(FILE* out, const uint8_t* data, unsigned size,
fwrite(&symbolList, 1, sizeof(symbolList), out);
fputc(0, out);
fwrite(dataName, 1, dataNameLength, out);
fwrite(sizeName, 1, sizeNameLength, out);
fwrite(startName, 1, startNameLength, out);
fwrite(endName, 1, endNameLength, out);
}
} // namespace
@ -138,7 +138,7 @@ main(int argc, const char** argv)
{
if (argc != 4) {
fprintf(stderr,
"usage: %s <input file> <data symbol name> <size symbol name>\n",
"usage: %s <input file> <start symbol name> <end symbol name>\n",
argv[0]);
return -1;
}

29
src/boot.cpp Normal file
View File

@ -0,0 +1,29 @@
#include "stdint.h"
#include "stdlib.h"
// since we don't link against libstdc++, we must implement some dummy
// functions:
extern "C" void __cxa_pure_virtual(void) { abort(); }
void operator delete(void*) { abort(); }
#ifdef __MINGW32__
# define EXPORT __declspec(dllexport)
# define SYMBOL(x) binary_classpath_jar_##x
#else
# define EXPORT __attribute__ ((visibility("default")))
# define SYMBOL(x) _binary_classpath_jar_##x
#endif
extern "C" {
extern const uint8_t SYMBOL(start)[];
extern const uint8_t SYMBOL(end)[];
EXPORT const uint8_t*
classpathJar(unsigned* size)
{
*size = SYMBOL(end) - SYMBOL(start);
return SYMBOL(start);
}
}

View File

@ -47,6 +47,19 @@ search(Thread* t, jstring name, object (*op)(Thread*, object),
}
}
void
enumerateThreads(Thread* t, Thread* x, object array, unsigned* index,
unsigned limit)
{
if (*index < limit) {
set(t, array, ArrayBody + (*index * BytesPerWord), x->javaThread);
if (x->peer) enumerateThreads(t, x->peer, array, index, limit);
if (x->child) enumerateThreads(t, x->child, array, index, limit);
}
}
} // namespace
extern "C" JNIEXPORT jstring JNICALL
@ -134,10 +147,10 @@ Java_java_lang_ClassLoader_defineClass
ENTER(t, Thread::ActiveState);
uint8_t* buffer = static_cast<uint8_t*>
(t->m->heap->allocate(length, false));
(t->m->heap->allocate(length));
memcpy(buffer, &byteArrayBody(t, *b, offset), length);
object c = parseClass(t, buffer, length);
t->m->heap->free(buffer, length, false);
t->m->heap->free(buffer, length);
return makeLocalReference(t, c);
}
@ -697,6 +710,36 @@ Java_java_lang_Thread_interrupt(Thread* t, jclass, jlong peer)
interrupt(t, reinterpret_cast<Thread*>(peer));
}
extern "C" JNIEXPORT jobject JNICALL
Java_java_lang_Thread_getTrace(Thread* t, jclass, jlong peer)
{
if (reinterpret_cast<Thread*>(peer) == t) {
return makeLocalReference(t, makeTrace(t));
} else {
return makeLocalReference
(t, t->m->processor->getStackTrace(t, reinterpret_cast<Thread*>(peer)));
}
}
extern "C" JNIEXPORT jint JNICALL
Java_java_lang_Thread_activeCount(Thread* t, jclass)
{
return t->m->liveCount;
}
extern "C" JNIEXPORT jint JNICALL
Java_java_lang_Thread_enumerate(Thread* t, jclass, jobjectArray array)
{
ACQUIRE_RAW(t, t->m->stateLock);
ENTER(t, Thread::ActiveState);
unsigned count = min(t->m->liveCount, objectArrayLength(t, *array));
unsigned index = 0;
enumerateThreads(t, t->m->rootThread, *array, &index, count);
return count;
}
extern "C" JNIEXPORT jlong JNICALL
Java_java_net_URL_00024ResourceInputStream_open
(Thread* t, jclass, jstring path)

File diff suppressed because it is too large Load Diff

View File

@ -745,9 +745,8 @@ register_(Context* c, int low, int high = NoRegister)
class MemoryValue: public Value {
public:
MemoryValue(int base, int offset, int index, unsigned scale,
TraceHandler* traceHandler):
value(base, offset, index, scale, traceHandler)
MemoryValue(int base, int offset, int index, unsigned scale):
value(base, offset, index, scale)
{ }
virtual OperandType type(Context*) { return Memory; }
@ -785,8 +784,8 @@ class MemoryValue: public Value {
class AbstractMemoryValue: public MemoryValue {
public:
AbstractMemoryValue(MyOperand* base, int offset, MyOperand* index,
unsigned scale, TraceHandler* traceHandler):
MemoryValue(NoRegister, offset, NoRegister, scale, traceHandler),
unsigned scale):
MemoryValue(NoRegister, offset, NoRegister, scale),
base_(base), index_(index)
{ }
@ -831,10 +830,10 @@ class AbstractMemoryValue: public MemoryValue {
AbstractMemoryValue*
memory(Context* c, MyOperand* base, int offset, MyOperand* index,
unsigned scale, TraceHandler* traceHandler)
unsigned scale)
{
return new (c->zone->allocate(sizeof(AbstractMemoryValue)))
AbstractMemoryValue(base, offset, index, scale, traceHandler);
AbstractMemoryValue(base, offset, index, scale);
}
class StackValue: public Value {
@ -1996,13 +1995,12 @@ class MyCompiler: public Compiler {
virtual Operand* memory(Operand* base,
int displacement = 0,
Operand* index = 0,
unsigned scale = 1,
TraceHandler* traceHandler = 0)
unsigned scale = 1)
{
MyOperand* result = operand
(&c, ::memory
(&c, static_cast<MyOperand*>(base), displacement,
static_cast<MyOperand*>(index), scale, traceHandler));
static_cast<MyOperand*>(index), scale));
appendMemory(&c, static_cast<MyOperand*>(base),
static_cast<MyOperand*>(index), result);

View File

@ -45,8 +45,8 @@ class Compiler {
virtual Operand* memory(Operand* base,
int displacement = 0,
Operand* index = 0,
unsigned scale = 1,
TraceHandler* traceHandler = 0) = 0;
unsigned scale = 1) = 0;
virtual Operand* stack() = 0;
virtual Operand* base() = 0;
virtual Operand* thread() = 0;

View File

@ -19,26 +19,19 @@ namespace {
void*
allocate(System* s, unsigned size)
{
void* p = s->tryAllocate(size, false);
void* p = s->tryAllocate(size);
if (p == 0) abort();
return p;
}
void
free(System* s, const void* p, unsigned size)
{
s->free(p, size, false);
}
const char*
append(System* s, unsigned* length, const char* a, const char* b,
append(System* s, const char* a, const char* b,
const char* c)
{
unsigned al = strlen(a);
unsigned bl = strlen(b);
unsigned cl = strlen(c);
*length = al + bl + cl;
char* p = static_cast<char*>(allocate(s, *length + 1));
char* p = static_cast<char*>(allocate(s, (al + bl + cl) + 1));
memcpy(p, a, al);
memcpy(p + al, b, bl);
memcpy(p + al + bl, c, cl + 1);
@ -46,11 +39,10 @@ append(System* s, unsigned* length, const char* a, const char* b,
}
const char*
copy(System* s, unsigned* length, const char* a)
copy(System* s, const char* a)
{
unsigned al = strlen(a);
*length = al;
char* p = static_cast<char*>(allocate(s, *length + 1));
char* p = static_cast<char*>(allocate(s, al + 1));
memcpy(p, a, al + 1);
return p;
}
@ -78,16 +70,15 @@ class Element {
class DirectoryElement: public Element {
public:
DirectoryElement(System* s, const char* name, unsigned nameLength):
s(s), name(name), nameLength(nameLength)
DirectoryElement(System* s, const char* name):
s(s), name(name)
{ }
virtual System::Region* find(const char* name) {
unsigned length;
const char* file = append(s, &length, this->name, "/", name);
const char* file = append(s, this->name, "/", name);
System::Region* region;
System::Status status = s->map(&region, file);
free(s, file, length + 1);
s->free(file);
if (s->success(status)) {
return region;
@ -97,21 +88,19 @@ class DirectoryElement: public Element {
}
virtual bool exists(const char* name) {
unsigned length;
const char* file = append(s, &length, this->name, "/", name);
const char* file = append(s, this->name, "/", name);
System::FileType type = s->identify(file);
free(s, file, length + 1);
s->free(file);
return type != System::DoesNotExist;
}
virtual void dispose() {
free(s, name, nameLength + 1);
free(s, this, sizeof(*this));
s->free(name);
s->free(this);
}
System* s;
const char* name;
unsigned nameLength;
};
class PointerRegion: public System::Region {
@ -131,7 +120,7 @@ class PointerRegion: public System::Region {
}
virtual void dispose() {
free(s, this, sizeof(*this));
s->free(this);
}
System* s;
@ -155,7 +144,7 @@ class DataRegion: public System::Region {
}
virtual void dispose() {
free(s, this, sizeof(*this) + length_);
s->free(this);
}
System* s;
@ -376,8 +365,8 @@ class JarIndex {
}
void dispose() {
free(s, nodes, sizeof(Node) * capacity);
free(s, this, sizeof(*this) + (sizeof(Node*) * capacity));
s->free(nodes);
s->free(this);
}
System* s;
@ -390,8 +379,8 @@ class JarIndex {
class JarElement: public Element {
public:
JarElement(System* s, const char* name, unsigned nameLength):
s(s), name(name), nameLength(nameLength), index(0)
JarElement(System* s, const char* name):
s(s), name(name), index(0)
{ }
virtual void init() {
@ -421,33 +410,32 @@ class JarElement: public Element {
}
virtual void dispose() {
free(s, name, nameLength + 1);
s->free(name);
if (index) {
index->dispose();
}
if (region) {
region->dispose();
}
free(s, this, sizeof(*this));
s->free(this);
}
System* s;
const char* name;
unsigned nameLength;
System::Region* region;
JarIndex* index;
};
class BuiltinElement: public JarElement {
public:
BuiltinElement(System* s, const char* name, unsigned nameLength):
JarElement(s, name, nameLength)
BuiltinElement(System* s, const char* name, const char* libraryName):
JarElement(s, name),
libraryName(libraryName ? copy(s, libraryName) : 0)
{ }
virtual void init() {
if (index == 0) {
System::Library* library;
if (s->success(s->load(&library, 0, false))) {
if (s->success(s->load(&library, libraryName, false))) {
void* p = library->resolve(name);
if (p) {
uint8_t* (*function)(unsigned*);
@ -461,14 +449,22 @@ class BuiltinElement: public JarElement {
index = JarIndex::open(s, region);
}
}
library->disposeAll();
}
}
}
virtual void dispose() {
library->disposeAll();
s->free(libraryName);
JarElement::dispose();
}
System::Library* library;
const char* libraryName;
};
Element*
parsePath(System* s, const char* path)
parsePath(System* s, const char* path, const char* bootLibrary)
{
class Tokenizer {
public:
@ -509,7 +505,7 @@ parsePath(System* s, const char* path)
name[token.length - 2] = 0;
e = new (allocate(s, sizeof(BuiltinElement)))
BuiltinElement(s, name, token.length - 2);
BuiltinElement(s, name, bootLibrary);
} else {
char* name = static_cast<char*>(allocate(s, token.length + 1));
memcpy(name, token.s, token.length);
@ -517,17 +513,16 @@ parsePath(System* s, const char* path)
switch (s->identify(name)) {
case System::File: {
e = new (allocate(s, sizeof(JarElement)))
JarElement(s, name, token.length);
e = new (allocate(s, sizeof(JarElement))) JarElement(s, name);
} break;
case System::Directory: {
e = new (allocate(s, sizeof(DirectoryElement)))
DirectoryElement(s, name, token.length);
DirectoryElement(s, name);
} break;
default: {
free(s, name, token.length + 1);
s->free(name);
e = 0;
} break;
}
@ -548,10 +543,10 @@ parsePath(System* s, const char* path)
class MyFinder: public Finder {
public:
MyFinder(System* system, const char* path):
MyFinder(System* system, const char* path, const char* bootLibrary):
system(system),
path_(parsePath(system, path)),
pathString(copy(system, &pathStringLength, path))
path_(parsePath(system, path, bootLibrary)),
pathString(copy(system, path))
{ }
virtual System::Region* find(const char* name) {
@ -585,14 +580,13 @@ class MyFinder: public Finder {
e = e->next;
t->dispose();
}
free(system, pathString, pathStringLength + 1);
free(system, this, sizeof(*this));
system->free(pathString);
system->free(this);
}
System* system;
Element* path_;
const char* pathString;
unsigned pathStringLength;
};
} // namespace
@ -600,9 +594,9 @@ class MyFinder: public Finder {
namespace vm {
Finder*
makeFinder(System* s, const char* path)
makeFinder(System* s, const char* path, const char* bootLibrary)
{
return new (allocate(s, sizeof(MyFinder))) MyFinder(s, path);
return new (allocate(s, sizeof(MyFinder))) MyFinder(s, path, bootLibrary);
}
} // namespace vm

View File

@ -27,7 +27,7 @@ class Finder {
};
Finder*
makeFinder(System* s, const char* path);
makeFinder(System* s, const char* path, const char* bootLibrary);
} // namespace vm

View File

@ -58,8 +58,8 @@ void assert(Context*, bool);
#endif
System* system(Context*);
void* tryAllocate(Context* c, unsigned size, bool executable);
void free(Context* c, const void* p, unsigned size, bool executable);
void* tryAllocate(Context* c, unsigned size);
void free(Context* c, const void* p, unsigned size);
inline void*
get(void* o, unsigned offsetInWords)
@ -78,7 +78,7 @@ set(void** o, void* value)
{
*o = reinterpret_cast<void*>
(reinterpret_cast<uintptr_t>(value)
| reinterpret_cast<uintptr_t>(*o) & (~PointerMask));
| (reinterpret_cast<uintptr_t>(*o) & (~PointerMask)));
}
inline void
@ -314,8 +314,7 @@ class Segment {
capacity_ = desired;
while (data == 0) {
data = static_cast<uintptr_t*>
(tryAllocate
(context, (footprint(capacity_)) * BytesPerWord, false));
(tryAllocate(context, (footprint(capacity_)) * BytesPerWord));
if (data == 0) {
if (capacity_ > minimum) {
@ -353,7 +352,7 @@ class Segment {
void replaceWith(Segment* s) {
if (data) {
free(context, data, (footprint(capacity())) * BytesPerWord, false);
free(context, data, (footprint(capacity())) * BytesPerWord);
}
data = s->data;
s->data = 0;
@ -404,7 +403,7 @@ class Segment {
}
void dispose() {
free(context, data, (footprint(capacity())) * BytesPerWord, false);
free(context, data, (footprint(capacity())) * BytesPerWord);
data = 0;
map = 0;
}
@ -783,7 +782,7 @@ free(Context* c, Fixie** fixies)
if (DebugFixies) {
fprintf(stderr, "free fixie %p\n", f);
}
free(c, f, f->totalSize(), false);
free(c, f, f->totalSize());
}
}
}
@ -1628,12 +1627,12 @@ collect(Context* c)
}
}
void* tryAllocate(Context* c, unsigned size, bool executable)
void* tryAllocate(Context* c, unsigned size)
{
ACQUIRE(c->lock);
if (size + c->count < c->limit) {
void* p = c->system->tryAllocate(size, executable);
void* p = c->system->tryAllocate(size);
if (p) {
c->count += size;
return p;
@ -1642,16 +1641,16 @@ void* tryAllocate(Context* c, unsigned size, bool executable)
return 0;
}
void free(Context* c, const void* p, unsigned size, bool executable) {
void free(Context* c, const void* p, unsigned size) {
ACQUIRE(c->lock);
expect(c->system, c->count >= size);
c->system->free(p, size, executable);
c->system->free(p);
c->count -= size;
}
void free_(Context* c, const void* p, unsigned size, bool executable) {
free(c, p, size, executable);
void free_(Context* c, const void* p, unsigned size) {
free(c, p, size);
}
class MyHeap: public Heap {
@ -1665,18 +1664,18 @@ class MyHeap: public Heap {
c.client = client;
}
virtual void* tryAllocate(unsigned size, bool executable) {
return ::tryAllocate(&c, size, executable);
virtual void* tryAllocate(unsigned size) {
return ::tryAllocate(&c, size);
}
virtual void* allocate(unsigned size, bool executable) {
void* p = ::tryAllocate(&c, size, executable);
virtual void* allocate(unsigned size) {
void* p = ::tryAllocate(&c, size);
expect(c.system, p);
return p;
}
virtual void free(const void* p, unsigned size, bool executable) {
free_(&c, p, size, executable);
virtual void free(const void* p, unsigned size) {
free_(&c, p, size);
}
virtual void collect(CollectionType type, unsigned incomingFootprint) {
@ -1690,16 +1689,15 @@ class MyHeap: public Heap {
bool objectMask, unsigned* totalInBytes)
{
*totalInBytes = Fixie::totalSize(sizeInWords, objectMask);
return (new (allocator->allocate(*totalInBytes, false))
return (new (allocator->allocate(*totalInBytes))
Fixie(sizeInWords, objectMask, &(c.fixies), false))->body();
}
virtual void* allocateImmortal(Allocator* allocator, unsigned sizeInWords,
bool executable, bool objectMask,
unsigned* totalInBytes)
bool objectMask, unsigned* totalInBytes)
{
*totalInBytes = Fixie::totalSize(sizeInWords, objectMask);
return (new (allocator->allocate(*totalInBytes, executable))
return (new (allocator->allocate(*totalInBytes))
Fixie(sizeInWords, objectMask, &(c.tenuredFixies), true))->body();
}
@ -1809,7 +1807,7 @@ class MyHeap: public Heap {
virtual void dispose() {
c.dispose();
assert(&c, c.count == 0);
c.system->free(this, sizeof(*this), false);
c.system->free(this);
}
Context c;
@ -1822,8 +1820,7 @@ namespace vm {
Heap*
makeHeap(System* system, unsigned limit)
{
return new (system->tryAllocate(sizeof(MyHeap), false))
MyHeap(system, limit);
return new (system->tryAllocate(sizeof(MyHeap))) MyHeap(system, limit);
}
} // namespace vm

View File

@ -60,8 +60,7 @@ class Heap: public Allocator {
virtual void* allocateFixed(Allocator* allocator, unsigned sizeInWords,
bool objectMask, unsigned* totalInBytes) = 0;
virtual void* allocateImmortal(Allocator* allocator, unsigned sizeInWords,
bool executable, bool objectMask,
unsigned* totalInBytes) = 0;
bool objectMask, unsigned* totalInBytes) = 0;
virtual bool needsMark(void* p) = 0;
virtual void mark(void* p, unsigned offset, unsigned count) = 0;
virtual void pad(void* p) = 0;

View File

@ -704,6 +704,50 @@ store(Thread* t, unsigned index)
BytesPerWord * 2);
}
ExceptionHandler*
findExceptionHandler(Thread* t, object method, unsigned ip)
{
PROTECT(t, method);
object eht = codeExceptionHandlerTable(t, methodCode(t, method));
if (eht) {
for (unsigned i = 0; i < exceptionHandlerTableLength(t, eht); ++i) {
ExceptionHandler* eh = exceptionHandlerTableBody(t, eht, i);
if (ip - 1 >= exceptionHandlerStart(eh)
and ip - 1 < exceptionHandlerEnd(eh))
{
object catchType = 0;
if (exceptionHandlerCatchType(eh)) {
object e = t->exception;
t->exception = 0;
PROTECT(t, e);
PROTECT(t, eht);
catchType = resolveClassInPool
(t, codePool(t, methodCode(t, method)),
exceptionHandlerCatchType(eh) - 1);
if (catchType) {
eh = exceptionHandlerTableBody(t, eht, i);
t->exception = e;
} else {
// can't find what we're supposed to catch - move on.
continue;
}
}
if (catchType == 0 or instanceOf(t, catchType, t->exception)) {
return eh;
}
}
}
}
return 0;
}
ExceptionHandler*
findExceptionHandler(Thread* t, int frame)
{
@ -1942,6 +1986,14 @@ interpret(Thread* t)
ip = (ip - 5) + static_cast<int32_t>(offset);
} goto loop;
case l2d: {
pushDouble(t, static_cast<double>(static_cast<int64_t>(popLong(t))));
} goto loop;
case l2f: {
pushFloat(t, static_cast<float>(static_cast<int64_t>(popLong(t))));
} goto loop;
case l2i: {
pushInt(t, static_cast<int32_t>(popLong(t)));
} goto loop;
@ -2999,6 +3051,11 @@ class MyProcessor: public Processor {
}
}
virtual object getStackTrace(Thread*, Thread*) {
// not implemented
return 0;
}
virtual void dispose(vm::Thread* t) {
t->m->heap->free(t, sizeof(Thread), false);
}

View File

@ -87,6 +87,35 @@ GetEnv(Machine* m, Thread** t, jint version)
}
}
jsize JNICALL
GetStringLength(Thread* t, jstring s)
{
ENTER(t, Thread::ActiveState);
return stringLength(t, *s);
}
const jchar* JNICALL
GetStringChars(Thread* t, jstring s, jboolean* isCopy)
{
ENTER(t, Thread::ActiveState);
jchar* chars = static_cast<jchar*>
(t->m->heap->allocate((stringLength(t, *s) + 1) * sizeof(jchar)));
stringChars(t, *s, chars);
if (isCopy) *isCopy = true;
return chars;
}
void JNICALL
ReleaseStringChars(Thread* t, jstring s, const jchar* chars)
{
ENTER(t, Thread::ActiveState);
t->m->heap->free(chars, (stringLength(t, *s) + 1) * sizeof(jchar));
}
jsize JNICALL
GetStringUTFLength(Thread* t, jstring s)
{
@ -101,7 +130,7 @@ GetStringUTFChars(Thread* t, jstring s, jboolean* isCopy)
ENTER(t, Thread::ActiveState);
char* chars = static_cast<char*>
(t->m->heap->allocate(stringLength(t, *s) + 1, false));
(t->m->heap->allocate(stringLength(t, *s) + 1));
stringChars(t, *s, chars);
if (isCopy) *isCopy = true;
@ -111,7 +140,9 @@ GetStringUTFChars(Thread* t, jstring s, jboolean* isCopy)
void JNICALL
ReleaseStringUTFChars(Thread* t, jstring s, const char* chars)
{
t->m->heap->free(chars, stringLength(t, *s) + 1, false);
ENTER(t, Thread::ActiveState);
t->m->heap->free(chars, stringLength(t, *s) + 1);
}
jsize JNICALL
@ -232,15 +263,13 @@ IsInstanceOf(Thread* t, jobject o, jclass c)
}
object
findMethod(Thread* t, object c, const char* name, const char* spec)
findMethod(Thread* t, jclass c, const char* name, const char* spec)
{
PROTECT(t, c);
object n = makeByteArray(t, "%s", name);
PROTECT(t, n);
object s = makeByteArray(t, "%s", spec);
return vm::findMethod(t, c, n, s);
return vm::findMethod(t, *c, n, s);
}
jmethodID JNICALL
@ -248,7 +277,7 @@ GetMethodID(Thread* t, jclass c, const char* name, const char* spec)
{
ENTER(t, Thread::ActiveState);
object method = findMethod(t, *c, name, spec);
object method = findMethod(t, c, name, spec);
if (UNLIKELY(t->exception)) return 0;
if (classFlags(t, *c) & ACC_INTERFACE) {
@ -278,7 +307,7 @@ GetStaticMethodID(Thread* t, jclass c, const char* name, const char* spec)
{
ENTER(t, Thread::ActiveState);
object method = findMethod(t, *c, name, spec);
object method = findMethod(t, c, name, spec);
if (UNLIKELY(t->exception)) return 0;
return methodOffset(t, method) + 1;
@ -289,11 +318,20 @@ getMethod(Thread* t, object o, jmethodID m)
{
if (m & InterfaceMethodID) {
return vectorBody(t, t->m->jniInterfaceTable, m & (~InterfaceMethodID));
} else if (m & NonVirtualMethodID) {
return arrayBody(t, classMethodTable(t, objectClass(t, o)),
m & (~NonVirtualMethodID));
} else {
return arrayBody(t, classVirtualTable(t, objectClass(t, o)), m - 1);
if (classVmFlags(t, objectClass(t, o)) & BootstrapFlag) {
PROTECT(t, o);
resolveClass(t, className(t, objectClass(t, o)));
if (UNLIKELY(t->exception)) return 0;
}
if (m & NonVirtualMethodID) {
return arrayBody(t, classMethodTable(t, objectClass(t, o)),
m & (~NonVirtualMethodID));
} else {
return arrayBody(t, classVirtualTable(t, objectClass(t, o)), m - 1);
}
}
}
@ -328,8 +366,9 @@ CallObjectMethodV(Thread* t, jobject o, jmethodID m, va_list a)
{
ENTER(t, Thread::ActiveState);
return makeLocalReference(t, t->m->processor->invokeList
(t, getMethod(t, *o, m), *o, true, a));
object method = getMethod(t, *o, m);
return makeLocalReference
(t, t->m->processor->invokeList(t, method, *o, true, a));
}
jobject JNICALL
@ -350,7 +389,8 @@ CallBooleanMethodV(Thread* t, jobject o, jmethodID m, va_list a)
{
ENTER(t, Thread::ActiveState);
object r = t->m->processor->invokeList(t, getMethod(t, *o, m), *o, true, a);
object method = getMethod(t, *o, m);
object r = t->m->processor->invokeList(t, method, *o, true, a);
return (t->exception ? false : (intValue(t, r) != 0));
}
@ -372,7 +412,8 @@ CallByteMethodV(Thread* t, jobject o, jmethodID m, va_list a)
{
ENTER(t, Thread::ActiveState);
object r = t->m->processor->invokeList(t, getMethod(t, *o, m), *o, true, a);
object method = getMethod(t, *o, m);
object r = t->m->processor->invokeList(t, method, *o, true, a);
return (t->exception ? 0 : intValue(t, r));
}
@ -394,7 +435,8 @@ CallCharMethodV(Thread* t, jobject o, jmethodID m, va_list a)
{
ENTER(t, Thread::ActiveState);
object r = t->m->processor->invokeList(t, getMethod(t, *o, m), *o, true, a);
object method = getMethod(t, *o, m);
object r = t->m->processor->invokeList(t, method, *o, true, a);
return (t->exception ? 0 : intValue(t, r));
}
@ -416,7 +458,8 @@ CallShortMethodV(Thread* t, jobject o, jmethodID m, va_list a)
{
ENTER(t, Thread::ActiveState);
object r = t->m->processor->invokeList(t, getMethod(t, *o, m), *o, true, a);
object method = getMethod(t, *o, m);
object r = t->m->processor->invokeList(t, method, *o, true, a);
return (t->exception ? 0 : intValue(t, r));
}
@ -438,7 +481,8 @@ CallIntMethodV(Thread* t, jobject o, jmethodID m, va_list a)
{
ENTER(t, Thread::ActiveState);
object r = t->m->processor->invokeList(t, getMethod(t, *o, m), *o, true, a);
object method = getMethod(t, *o, m);
object r = t->m->processor->invokeList(t, method, *o, true, a);
return (t->exception ? 0 : intValue(t, r));
}
@ -460,7 +504,8 @@ CallLongMethodV(Thread* t, jobject o, jmethodID m, va_list a)
{
ENTER(t, Thread::ActiveState);
object r = t->m->processor->invokeList(t, getMethod(t, *o, m), *o, true, a);
object method = getMethod(t, *o, m);
object r = t->m->processor->invokeList(t, method, *o, true, a);
return (t->exception ? 0 : longValue(t, r));
}
@ -482,7 +527,8 @@ CallFloatMethodV(Thread* t, jobject o, jmethodID m, va_list a)
{
ENTER(t, Thread::ActiveState);
object r = t->m->processor->invokeList(t, getMethod(t, *o, m), *o, true, a);
object method = getMethod(t, *o, m);
object r = t->m->processor->invokeList(t, method, *o, true, a);
return (t->exception ? 0 : bitsToFloat(intValue(t, r)));
}
@ -504,7 +550,8 @@ CallDoubleMethodV(Thread* t, jobject o, jmethodID m, va_list a)
{
ENTER(t, Thread::ActiveState);
object r = t->m->processor->invokeList(t, getMethod(t, *o, m), *o, true, a);
object method = getMethod(t, *o, m);
object r = t->m->processor->invokeList(t, method, *o, true, a);
return (t->exception ? 0 : bitsToDouble(longValue(t, r)));
}
@ -526,7 +573,8 @@ CallVoidMethodV(Thread* t, jobject o, jmethodID m, va_list a)
{
ENTER(t, Thread::ActiveState);
t->m->processor->invokeList(t, getMethod(t, *o, m), *o, true, a);
object method = getMethod(t, *o, m);
t->m->processor->invokeList(t, method, *o, true, a);
}
void JNICALL
@ -541,9 +589,16 @@ CallVoidMethod(Thread* t, jobject o, jmethodID m, ...)
}
inline object
getStaticMethod(Thread* t, object class_, jmethodID m)
getStaticMethod(Thread* t, object c, jmethodID m)
{
return arrayBody(t, classMethodTable(t, class_), m - 1);
if (classVmFlags(t, c) & BootstrapFlag) {
PROTECT(t, c);
resolveClass(t, className(t, c));
if (UNLIKELY(t->exception)) return 0;
}
return arrayBody(t, classMethodTable(t, c), m - 1);
}
jobject JNICALL
@ -772,13 +827,13 @@ CallStaticVoidMethod(Thread* t, jclass c, jmethodID m, ...)
}
object
findField(Thread* t, object class_, const char* name, const char* spec)
findField(Thread* t, jclass c, const char* name, const char* spec)
{
object n = makeByteArray(t, "%s", name);
PROTECT(t, n);
object s = makeByteArray(t, "%s", spec);
return vm::findField(t, class_, n, s);
return vm::findField(t, *c, n, s);
}
jfieldID JNICALL
@ -786,7 +841,7 @@ GetFieldID(Thread* t, jclass c, const char* name, const char* spec)
{
ENTER(t, Thread::ActiveState);
object field = findField(t, *c, name, spec);
object field = findField(t, c, name, spec);
if (UNLIKELY(t->exception)) return 0;
return fieldOffset(t, field);
@ -797,7 +852,7 @@ GetStaticFieldID(Thread* t, jclass c, const char* name, const char* spec)
{
ENTER(t, Thread::ActiveState);
object field = findField(t, *c, name, spec);
object field = findField(t, c, name, spec);
if (UNLIKELY(t->exception)) return 0;
return fieldOffset(t, field);
@ -1122,7 +1177,7 @@ NewGlobalRef(Thread* t, jobject o)
ACQUIRE(t, t->m->referenceLock);
if (o) {
Reference* r = new (t->m->heap->allocate(sizeof(Reference), false))
Reference* r = new (t->m->heap->allocate(sizeof(Reference)))
Reference(*o, &(t->m->jniReferences));
return &(r->target);
@ -1267,7 +1322,7 @@ GetBooleanArrayElements(Thread* t, jbooleanArray array, jboolean* isCopy)
ENTER(t, Thread::ActiveState);
unsigned size = booleanArrayLength(t, *array) * sizeof(jboolean);
jboolean* p = static_cast<jboolean*>(t->m->heap->allocate(size, false));
jboolean* p = static_cast<jboolean*>(t->m->heap->allocate(size));
if (size) {
memcpy(p, &booleanArrayBody(t, *array, 0), size);
}
@ -1285,7 +1340,7 @@ GetByteArrayElements(Thread* t, jbyteArray array, jboolean* isCopy)
ENTER(t, Thread::ActiveState);
unsigned size = byteArrayLength(t, *array) * sizeof(jbyte);
jbyte* p = static_cast<jbyte*>(t->m->heap->allocate(size, false));
jbyte* p = static_cast<jbyte*>(t->m->heap->allocate(size));
if (size) {
memcpy(p, &byteArrayBody(t, *array, 0), size);
}
@ -1303,7 +1358,7 @@ GetCharArrayElements(Thread* t, jcharArray array, jboolean* isCopy)
ENTER(t, Thread::ActiveState);
unsigned size = charArrayLength(t, *array) * sizeof(jchar);
jchar* p = static_cast<jchar*>(t->m->heap->allocate(size, false));
jchar* p = static_cast<jchar*>(t->m->heap->allocate(size));
if (size) {
memcpy(p, &charArrayBody(t, *array, 0), size);
}
@ -1321,7 +1376,7 @@ GetShortArrayElements(Thread* t, jshortArray array, jboolean* isCopy)
ENTER(t, Thread::ActiveState);
unsigned size = shortArrayLength(t, *array) * sizeof(jshort);
jshort* p = static_cast<jshort*>(t->m->heap->allocate(size, false));
jshort* p = static_cast<jshort*>(t->m->heap->allocate(size));
if (size) {
memcpy(p, &shortArrayBody(t, *array, 0), size);
}
@ -1339,7 +1394,7 @@ GetIntArrayElements(Thread* t, jintArray array, jboolean* isCopy)
ENTER(t, Thread::ActiveState);
unsigned size = intArrayLength(t, *array) * sizeof(jint);
jint* p = static_cast<jint*>(t->m->heap->allocate(size, false));
jint* p = static_cast<jint*>(t->m->heap->allocate(size));
if (size) {
memcpy(p, &intArrayBody(t, *array, 0), size);
}
@ -1357,7 +1412,7 @@ GetLongArrayElements(Thread* t, jlongArray array, jboolean* isCopy)
ENTER(t, Thread::ActiveState);
unsigned size = longArrayLength(t, *array) * sizeof(jlong);
jlong* p = static_cast<jlong*>(t->m->heap->allocate(size, false));
jlong* p = static_cast<jlong*>(t->m->heap->allocate(size));
if (size) {
memcpy(p, &longArrayBody(t, *array, 0), size);
}
@ -1375,7 +1430,7 @@ GetFloatArrayElements(Thread* t, jfloatArray array, jboolean* isCopy)
ENTER(t, Thread::ActiveState);
unsigned size = floatArrayLength(t, *array) * sizeof(jfloat);
jfloat* p = static_cast<jfloat*>(t->m->heap->allocate(size, false));
jfloat* p = static_cast<jfloat*>(t->m->heap->allocate(size));
if (size) {
memcpy(p, &floatArrayBody(t, *array, 0), size);
}
@ -1393,7 +1448,7 @@ GetDoubleArrayElements(Thread* t, jdoubleArray array, jboolean* isCopy)
ENTER(t, Thread::ActiveState);
unsigned size = doubleArrayLength(t, *array) * sizeof(jdouble);
jdouble* p = static_cast<jdouble*>(t->m->heap->allocate(size, false));
jdouble* p = static_cast<jdouble*>(t->m->heap->allocate(size));
if (size) {
memcpy(p, &doubleArrayBody(t, *array, 0), size);
}
@ -1420,7 +1475,7 @@ ReleaseBooleanArrayElements(Thread* t, jbooleanArray array, jboolean* p,
}
if (mode == 0 or mode == JNI_ABORT) {
t->m->heap->free(p, size, false);
t->m->heap->free(p, size);
}
}
@ -1438,7 +1493,7 @@ ReleaseByteArrayElements(Thread* t, jbyteArray array, jbyte* p, jint mode)
}
if (mode == 0 or mode == JNI_ABORT) {
t->m->heap->free(p, size, false);
t->m->heap->free(p, size);
}
}
@ -1456,7 +1511,7 @@ ReleaseCharArrayElements(Thread* t, jcharArray array, jchar* p, jint mode)
}
if (mode == 0 or mode == JNI_ABORT) {
t->m->heap->free(p, size, false);
t->m->heap->free(p, size);
}
}
@ -1474,7 +1529,7 @@ ReleaseShortArrayElements(Thread* t, jshortArray array, jshort* p, jint mode)
}
if (mode == 0 or mode == JNI_ABORT) {
t->m->heap->free(p, size, false);
t->m->heap->free(p, size);
}
}
@ -1492,7 +1547,7 @@ ReleaseIntArrayElements(Thread* t, jintArray array, jint* p, jint mode)
}
if (mode == 0 or mode == JNI_ABORT) {
t->m->heap->free(p, size, false);
t->m->heap->free(p, size);
}
}
@ -1510,7 +1565,7 @@ ReleaseLongArrayElements(Thread* t, jlongArray array, jlong* p, jint mode)
}
if (mode == 0 or mode == JNI_ABORT) {
t->m->heap->free(p, size, false);
t->m->heap->free(p, size);
}
}
@ -1528,7 +1583,7 @@ ReleaseFloatArrayElements(Thread* t, jfloatArray array, jfloat* p, jint mode)
}
if (mode == 0 or mode == JNI_ABORT) {
t->m->heap->free(p, size, false);
t->m->heap->free(p, size);
}
}
@ -1547,7 +1602,7 @@ ReleaseDoubleArrayElements(Thread* t, jdoubleArray array, jdouble* p,
}
if (mode == 0 or mode == JNI_ABORT) {
t->m->heap->free(p, size, false);
t->m->heap->free(p, size);
}
}
@ -1781,9 +1836,13 @@ GetJavaVM(Thread* t, Machine** m)
jboolean JNICALL
IsSameObject(Thread* t, jobject a, jobject b)
{
ENTER(t, Thread::ActiveState);
if (a and b) {
ENTER(t, Thread::ActiveState);
return *a == *b;
return *a == *b;
} else {
return a == b;
}
}
struct JDK1_1InitArgs {
@ -1826,6 +1885,9 @@ populateJNITables(JavaVMVTable* vmTable, JNIEnvVTable* envTable)
memset(envTable, 0, sizeof(JNIEnvVTable));
envTable->GetStringLength = ::GetStringLength;
envTable->GetStringChars = ::GetStringChars;
envTable->ReleaseStringChars = ::ReleaseStringChars;
envTable->GetStringUTFLength = ::GetStringUTFLength;
envTable->GetStringUTFChars = ::GetStringUTFChars;
envTable->ReleaseStringUTFChars = ::ReleaseStringUTFChars;
@ -1988,28 +2050,38 @@ JNI_GetDefaultJavaVMInitArgs(void* args)
return 0;
}
#define BUILTINS_PROPERTY "vm.builtins"
#define BUILTINS_PROPERTY "avian.builtins"
#define BOOTSTRAP_PROPERTY "avian.bootstrap"
extern "C" JNIEXPORT jint JNICALL
JNI_CreateJavaVM(Machine** m, Thread** t, void* args)
{
JDK1_1InitArgs* a = static_cast<JDK1_1InitArgs*>(args);
System* s = makeSystem();
Heap* h = makeHeap(s, a->maxHeapSize);
Finder* f = makeFinder(s, a->classpath);
Processor* p = makeProcessor(s, h);
*m = new (h->allocate(sizeof(Machine), false)) Machine(s, h, f, p);
const char* builtins = 0;
const char* bootLibrary = 0;
if (a->properties) {
for (const char** p = a->properties; *p; ++p) {
if (strncmp(*p, BUILTINS_PROPERTY "=", sizeof(BUILTINS_PROPERTY)) == 0) {
(*m)->builtins = (*p) + sizeof(BUILTINS_PROPERTY);
if (strncmp(*p, BUILTINS_PROPERTY "=",
sizeof(BUILTINS_PROPERTY)) == 0)
{
builtins = (*p) + sizeof(BUILTINS_PROPERTY);
} else if (strncmp(*p, BOOTSTRAP_PROPERTY "=",
sizeof(BOOTSTRAP_PROPERTY)) == 0)
{
bootLibrary = (*p) + sizeof(BOOTSTRAP_PROPERTY);
}
}
}
System* s = makeSystem();
Heap* h = makeHeap(s, a->maxHeapSize);
Finder* f = makeFinder(s, a->classpath, bootLibrary);
Processor* p = makeProcessor(s, h);
*m = new (h->allocate(sizeof(Machine)))
Machine(s, h, f, p, bootLibrary, builtins);
*t = p->makeThread(*m, 0, 0);
enter(*t, Thread::ActiveState);

View File

@ -22,15 +22,9 @@ namespace {
bool
find(Thread* t, Thread* o)
{
if (t == o) return true;
for (Thread* p = t->peer; p; p = p->peer) {
if (p == o) return true;
}
if (t->child) return find(t->child, o);
return false;
return (t == o)
or (t->peer and find(t->peer, o))
or (t->child and find(t->child, o));
}
void
@ -48,11 +42,7 @@ count(Thread* t, Thread* o)
unsigned c = 0;
if (t != o) ++ c;
for (Thread* p = t->peer; p; p = p->peer) {
c += count(p, o);
}
if (t->peer) c += count(t->peer, o);
if (t->child) c += count(t->child, o);
return c;
@ -62,12 +52,8 @@ Thread**
fill(Thread* t, Thread* o, Thread** array)
{
if (t != o) *(array++) = t;
for (Thread* p = t->peer; p; p = p->peer) {
array = fill(p, o, array);
}
if (t->child) array = fill(t->child, o, array);
if (t->peer) fill(t->peer, o, array);
if (t->child) fill(t->child, o, array);
return array;
}
@ -492,15 +478,22 @@ void
postCollect(Thread* t)
{
#ifdef VM_STRESS
t->m->heap->free(t->defaultHeap, Thread::HeapSizeInBytes, false);
t->m->heap->free(t->defaultHeap, Thread::HeapSizeInBytes);
t->defaultHeap = static_cast<uintptr_t*>
(t->m->heap->allocate(Thread::HeapSizeInBytes, false));
(t->m->heap->allocate(Thread::HeapSizeInBytes));
#endif
t->heap = t->defaultHeap;
t->heapOffset = 0;
t->heapIndex = 0;
if (t->backupHeap) {
t->m->heap->free
(t->backupHeap, t->backupHeapSizeInWords * BytesPerWord);
t->backupHeapIndex = 0;
t->backupHeapSizeInWords = 0;
}
for (Thread* c = t->child; c; c = c->peer) {
postCollect(c);
}
@ -660,8 +653,7 @@ parsePool(Thread* t, Stream& s)
PROTECT(t, pool);
if (count) {
uint32_t* index = static_cast<uint32_t*>
(t->m->heap->allocate(count * 4, false));
uint32_t* index = static_cast<uint32_t*>(t->m->heap->allocate(count * 4));
for (unsigned i = 0; i < count; ++i) {
index[i] = s.position();
@ -707,7 +699,7 @@ parsePool(Thread* t, Stream& s)
i += parsePoolEntry(t, s, index, pool, i);
}
t->m->heap->free(index, count * 4, false);
t->m->heap->free(index, count * 4);
s.setPosition(end);
}
@ -861,7 +853,7 @@ parseFieldTable(Thread* t, Stream& s, object class_, object pool)
if (flags & ACC_STATIC) {
unsigned size = fieldSize(t, code);
unsigned excess = staticOffset % size;
unsigned excess = (staticOffset % size) % BytesPerWord;
if (excess) {
staticOffset += BytesPerWord - excess;
}
@ -878,7 +870,7 @@ parseFieldTable(Thread* t, Stream& s, object class_, object pool)
abort(t); // todo: handle non-static field initializers
}
unsigned excess = memberOffset % fieldSize(t, code);
unsigned excess = (memberOffset % fieldSize(t, code)) % BytesPerWord;
if (excess) {
memberOffset += BytesPerWord - excess;
}
@ -1294,7 +1286,9 @@ parseMethodTable(Thread* t, Stream& s, object class_, object pool)
object method = arrayBody(t, ivtable, j);
method = hashMapFind
(t, virtualMap, method, methodHash, methodEqual);
assert(t, method);
// note that method may be null in the case of an abstract
// class
set(t, vtable, ArrayBody + (j * BytesPerWord), method);
}
@ -1630,7 +1624,7 @@ class HeapClient: public Heap::Client {
}
void dispose() {
m->heap->free(this, sizeof(*this), false);
m->heap->free(this, sizeof(*this));
}
private:
@ -1642,10 +1636,11 @@ class HeapClient: public Heap::Client {
namespace vm {
Machine::Machine(System* system, Heap* heap, Finder* finder,
Processor* processor):
Processor* processor, const char* bootLibrary,
const char* builtins):
vtable(&javaVMVTable),
system(system),
heapClient(new (heap->allocate(sizeof(HeapClient), false))
heapClient(new (heap->allocate(sizeof(HeapClient)))
HeapClient(this)),
heap(heap),
finder(finder),
@ -1653,7 +1648,7 @@ Machine::Machine(System* system, Heap* heap, Finder* finder,
rootThread(0),
exclusive(0),
jniReferences(0),
builtins(0),
builtins(builtins),
activeCount(0),
liveCount(0),
fixedFootprint(0),
@ -1686,7 +1681,7 @@ Machine::Machine(System* system, Heap* heap, Finder* finder,
not system->success(system->make(&heapLock)) or
not system->success(system->make(&classLock)) or
not system->success(system->make(&referenceLock)) or
not system->success(system->load(&libraries, 0, false)))
not system->success(system->load(&libraries, bootLibrary, false)))
{
system->abort();
}
@ -1708,16 +1703,16 @@ Machine::dispose()
for (Reference* r = jniReferences; r;) {
Reference* tmp = r;
r = r->next;
heap->free(tmp, sizeof(*tmp), false);
heap->free(tmp, sizeof(*tmp));
}
for (unsigned i = 0; i < heapPoolIndex; ++i) {
heap->free(heapPool[i], Thread::HeapSizeInBytes, false);
heap->free(heapPool[i], Thread::HeapSizeInBytes);
}
static_cast<HeapClient*>(heapClient)->dispose();
heap->free(this, sizeof(*this), false);
heap->free(this, sizeof(*this));
}
Thread::Thread(Machine* m, object javaThread, Thread* parent):
@ -1736,8 +1731,11 @@ Thread::Thread(Machine* m, object javaThread, Thread* parent):
protector(0),
runnable(this),
defaultHeap(static_cast<uintptr_t*>
(m->heap->allocate(HeapSizeInBytes, false))),
heap(defaultHeap)
(m->heap->allocate(HeapSizeInBytes))),
heap(defaultHeap),
backupHeap(0),
backupHeapIndex(0),
backupHeapSizeInWords(0)
#ifdef VM_STRESS
, stress(false)
#endif // VM_STRESS
@ -1873,7 +1871,7 @@ Thread::dispose()
systemThread->dispose();
}
m->heap->free(defaultHeap, Thread::HeapSizeInBytes, false);
m->heap->free(defaultHeap, Thread::HeapSizeInBytes);
m->processor->dispose(this);
}
@ -2033,13 +2031,23 @@ allocate2(Thread* t, unsigned sizeInBytes, bool objectMask)
(t, t->m->heap,
ceiling(sizeInBytes, BytesPerWord) > Thread::HeapSizeInWords ?
Machine::FixedAllocation : Machine::MovableAllocation,
sizeInBytes, false, objectMask);
sizeInBytes, objectMask);
}
object
allocate3(Thread* t, Allocator* allocator, Machine::AllocationType type,
unsigned sizeInBytes, bool executable, bool objectMask)
unsigned sizeInBytes, bool objectMask)
{
if (t->backupHeap) {
expect(t, t->backupHeapIndex + ceiling(sizeInBytes, BytesPerWord)
<= t->backupHeapSizeInWords);
object o = reinterpret_cast<object>(t->backupHeap + t->backupHeapIndex);
t->backupHeapIndex += ceiling(sizeInBytes, BytesPerWord);
cast<object>(o, 0) = 0;
return o;
}
ACQUIRE_RAW(t, t->m->stateLock);
while (t->m->exclusive and t->m->exclusive != t) {
@ -2055,12 +2063,12 @@ allocate3(Thread* t, Allocator* allocator, Machine::AllocationType type,
t->heap = 0;
}
} else if (t->heapIndex + ceiling(sizeInBytes, BytesPerWord)
>= Thread::HeapSizeInWords)
> Thread::HeapSizeInWords)
{
t->heap = 0;
if (t->m->heapPoolIndex < Machine::HeapPoolSize) {
t->heap = static_cast<uintptr_t*>
(t->m->heap->tryAllocate(Thread::HeapSizeInBytes, false));
(t->m->heap->tryAllocate(Thread::HeapSizeInBytes));
if (t->heap) {
t->m->heapPool[t->m->heapPoolIndex++] = t->heap;
t->heapOffset += t->heapIndex;
@ -2098,8 +2106,7 @@ allocate3(Thread* t, Allocator* allocator, Machine::AllocationType type,
unsigned total;
object o = static_cast<object>
(t->m->heap->allocateImmortal
(allocator, ceiling(sizeInBytes, BytesPerWord),
executable, objectMask, &total));
(allocator, ceiling(sizeInBytes, BytesPerWord), objectMask, &total));
cast<uintptr_t>(o, 0) = FixedMark;
@ -2150,6 +2157,24 @@ stringChars(Thread* t, object string, char* chars)
chars[stringLength(t, string)] = 0;
}
void
stringChars(Thread* t, object string, uint16_t* chars)
{
object data = stringData(t, string);
if (objectClass(t, data)
== arrayBody(t, t->m->types, Machine::ByteArrayType))
{
for (unsigned i = 0; i < stringLength(t, string); ++i) {
chars[i] = byteArrayBody(t, data, stringOffset(t, string) + i);
}
} else {
memcpy(chars,
&charArrayBody(t, data, stringOffset(t, string)),
stringLength(t, string) * sizeof(uint16_t));
}
chars[stringLength(t, string)] = 0;
}
bool
isAssignableFrom(Thread* t, object a, object b)
{
@ -2727,7 +2752,7 @@ collect(Thread* t, Heap::CollectionType type)
killZombies(t, m->rootThread);
for (unsigned i = 0; i < m->heapPoolIndex; ++i) {
m->heap->free(m->heapPool[i], Thread::HeapSizeInBytes, false);
m->heap->free(m->heapPool[i], Thread::HeapSizeInBytes);
}
m->heapPoolIndex = 0;
@ -2818,7 +2843,7 @@ makeTrace(Thread* t, Processor::StackWalker* walker)
}
object
makeTrace(Thread* t)
makeTrace(Thread* t, Thread* target)
{
class Visitor: public Processor::StackVisitor {
public:
@ -2833,7 +2858,7 @@ makeTrace(Thread* t)
object trace;
} v(t);
t->m->processor->walkStack(t, &v);
t->m->processor->walkStack(target, &v);
return v.trace ? v.trace : makeArray(t, 0, true);
}

View File

@ -1121,7 +1121,8 @@ class Machine {
ImmortalAllocation
};
Machine(System* system, Heap* heap, Finder* finder, Processor* processor);
Machine(System* system, Heap* heap, Finder* finder, Processor* processor,
const char* bootLibrary, const char* builtins);
~Machine() {
dispose();
@ -1287,6 +1288,9 @@ class Thread {
Runnable runnable;
uintptr_t* defaultHeap;
uintptr_t* heap;
uintptr_t* backupHeap;
unsigned backupHeapIndex;
unsigned backupHeapSizeInWords;
#ifdef VM_STRESS
bool stress;
#endif // VM_STRESS
@ -1327,7 +1331,7 @@ dispose(Thread* t, Reference* r)
if (r->next) {
r->next->handle = r->handle;
}
t->m->heap->free(r, sizeof(*r), false);
t->m->heap->free(r, sizeof(*r));
}
void
@ -1425,12 +1429,26 @@ expect(Thread* t, bool v)
expect(t->m->system, v);
}
inline void
ensure(Thread* t, unsigned sizeInBytes)
{
if (t->heapIndex + ceiling(sizeInBytes, BytesPerWord)
> Thread::HeapSizeInWords)
{
expect(t, t->backupHeap == 0);
t->backupHeap = static_cast<uintptr_t*>
(t->m->heap->allocate(pad(sizeInBytes)));
t->backupHeapIndex = 0;
t->backupHeapSizeInWords = ceiling(sizeInBytes, BytesPerWord);
}
}
object
allocate2(Thread* t, unsigned sizeInBytes, bool objectMask);
object
allocate3(Thread* t, Allocator* allocator, Machine::AllocationType type,
unsigned sizeInBytes, bool executable, bool objectMask);
unsigned sizeInBytes, bool objectMask);
inline object
allocateSmall(Thread* t, unsigned sizeInBytes)
@ -1447,7 +1465,7 @@ allocate(Thread* t, unsigned sizeInBytes, bool objectMask)
stress(t);
if (UNLIKELY(t->heapIndex + ceiling(sizeInBytes, BytesPerWord)
>= Thread::HeapSizeInWords
> Thread::HeapSizeInWords
or t->m->exclusive))
{
return allocate2(t, sizeInBytes, objectMask);
@ -1484,7 +1502,7 @@ setObjectClass(Thread*, object o, object value)
cast<object>(o, 0)
= reinterpret_cast<object>
(reinterpret_cast<uintptr_t>(value)
| reinterpret_cast<uintptr_t>(cast<object>(o, 0)) & (~PointerMask));
| (reinterpret_cast<uintptr_t>(cast<object>(o, 0)) & (~PointerMask)));
}
object&
@ -1526,7 +1544,13 @@ object
makeTrace(Thread* t, Processor::StackWalker* walker);
object
makeTrace(Thread* t);
makeTrace(Thread* t, Thread* target);
inline object
makeTrace(Thread* t)
{
return makeTrace(t, t);
}
inline object
makeRuntimeException(Thread* t, object message)
@ -1702,6 +1726,9 @@ makeString(Thread* t, const char* format, ...);
void
stringChars(Thread* t, object string, char* chars);
void
stringChars(Thread* t, object string, uint16_t* chars);
bool
isAssignableFrom(Thread* t, object a, object b);
@ -2231,6 +2258,7 @@ inline object
makeSingleton(Thread* t, unsigned count)
{
object o = makeSingleton(t, count + singletonMaskSize(count), true);
assert(t, singletonLength(t, o) == count + singletonMaskSize(t, o));
if (count) {
singletonMask(t, o)[0] = 1;
}

View File

@ -14,37 +14,12 @@
#include "stdint.h"
#include "jni.h"
// since we don't link against libstdc++, we must implement some dummy
// functions:
extern "C" void __cxa_pure_virtual(void) { abort(); }
void operator delete(void*) { abort(); }
#ifdef __MINGW32__
# define PATH_SEPARATOR ';'
# define EXPORT __declspec(dllexport)
# define SYMBOL(x) binary_classpath_jar_##x
#else
# define PATH_SEPARATOR ':'
# define EXPORT __attribute__ ((visibility("default")))
# define SYMBOL(x) _binary_classpath_jar_##x
#endif
#define BOOT_CLASSPATH "[classpathJar]"
extern "C" {
extern const uint8_t SYMBOL(start)[];
extern const uint8_t SYMBOL(size)[];
EXPORT const uint8_t*
classpathJar(unsigned* size)
{
*size = reinterpret_cast<uintptr_t>(SYMBOL(size));
return SYMBOL(start);
}
}
#ifdef JNI_VERSION_1_6
// todo: use JavaVMInitArgs instead
typedef struct JDK1_1InitArgs {
@ -121,14 +96,25 @@ main(int ac, const char** av)
vmArgs.classpath = classpath;
#endif
const char* properties[propertyCount + 1];
properties[propertyCount] = 0;
#ifdef BOOT_LIBRARY
const int BootPropertyCount = 1;
#else
const int BootPropertyCount = 0;
#endif
const char* properties[propertyCount + BootPropertyCount + 1];
properties[propertyCount + BootPropertyCount] = 0;
for (int i = 1; i < ac; ++i) {
if (strncmp(av[i], "-D", 2) == 0) {
properties[--propertyCount] = av[i] + 2;
}
}
#ifdef BOOT_LIBRARY
properties[propertyCount + BootPropertyCount - 1]
= "avian.bootstrap=" BOOT_LIBRARY;
#endif
vmArgs.properties = const_cast<char**>(properties);
if (class_ == 0) {

View File

@ -35,9 +35,6 @@ using namespace vm;
namespace {
System::SignalHandler* segFaultHandler = 0;
struct sigaction oldSegFaultHandler;
class MutexResource {
public:
MutexResource(pthread_mutex_t& m): m(&m) {
@ -52,12 +49,22 @@ class MutexResource {
pthread_mutex_t* m;
};
const int InterruptSignal = SIGUSR2;
const int VisitSignal = SIGUSR1;
#ifdef __APPLE__
const int SegFaultSignal = SIGBUS;
#else
const int SegFaultSignal = SIGSEGV;
#endif
const int InterruptSignal = SIGUSR2;
const unsigned VisitSignalIndex = 0;
const unsigned SegFaultSignalIndex = 1;
const unsigned InterruptSignalIndex = 2;
class MySystem;
MySystem* system;
const int signals[] = { VisitSignal, SegFaultSignal, InterruptSignal };
#ifdef __x86_64__
# define IP_REGISTER(context) (context->uc_mcontext.gregs[REG_RIP])
@ -65,10 +72,17 @@ const int SegFaultSignal = SIGSEGV;
# define STACK_REGISTER(context) (context->uc_mcontext.gregs[REG_RSP])
# define THREAD_REGISTER(context) (context->uc_mcontext.gregs[REG_RBX])
#elif defined __APPLE__
# define IP_REGISTER(context) (context->uc_mcontext->__ss.__eip)
# define BASE_REGISTER(context) (context->uc_mcontext->__ss.__ebp)
# define STACK_REGISTER(context) (context->uc_mcontext->__ss.__esp)
# define THREAD_REGISTER(context) (context->uc_mcontext->__ss.__ebx)
# if __DARWIN_UNIX03 && defined(_STRUCT_X86_EXCEPTION_STATE32)
# define IP_REGISTER(context) (context->uc_mcontext->__ss.__eip)
# define BASE_REGISTER(context) (context->uc_mcontext->__ss.__ebp)
# define STACK_REGISTER(context) (context->uc_mcontext->__ss.__esp)
# define THREAD_REGISTER(context) (context->uc_mcontext->__ss.__ebx)
# else
# define IP_REGISTER(context) (context->uc_mcontext->ss.eip)
# define BASE_REGISTER(context) (context->uc_mcontext->ss.ebp)
# define STACK_REGISTER(context) (context->uc_mcontext->ss.esp)
# define THREAD_REGISTER(context) (context->uc_mcontext->ss.ebx)
# endif
#elif defined __i386__
# define IP_REGISTER(context) (context->uc_mcontext.gregs[REG_EIP])
# define BASE_REGISTER(context) (context->uc_mcontext.gregs[REG_EBP])
@ -79,41 +93,7 @@ const int SegFaultSignal = SIGSEGV;
#endif
void
handleSignal(int signal, siginfo_t* info, void* context)
{
if (signal == SegFaultSignal) {
ucontext_t* c = static_cast<ucontext_t*>(context);
void* ip = reinterpret_cast<void*>(IP_REGISTER(c));
void* base = reinterpret_cast<void*>(BASE_REGISTER(c));
void* stack = reinterpret_cast<void*>(STACK_REGISTER(c));
void* thread = reinterpret_cast<void*>(THREAD_REGISTER(c));
bool jump = segFaultHandler->handleSignal
(&ip, &base, &stack, &thread);
if (jump) {
// I'd like to use setcontext here (and get rid of the
// sigprocmask call), but it doesn't work on my Linux x86_64
// system, and I can't tell from the documentation if it's even
// supposed to work.
sigset_t set;
sigemptyset(&set);
sigaddset(&set, SegFaultSignal);
sigprocmask(SIG_UNBLOCK, &set, 0);
vmJump(ip, base, stack, thread);
} else if (oldSegFaultHandler.sa_flags & SA_SIGINFO) {
oldSegFaultHandler.sa_sigaction(signal, info, context);
} else if (oldSegFaultHandler.sa_handler) {
oldSegFaultHandler.sa_handler(signal);
} else {
abort();
}
}
}
handleSignal(int signal, siginfo_t* info, void* context);
void*
run(void* r)
@ -125,7 +105,7 @@ run(void* r)
void*
allocate(System* s, unsigned size)
{
void* p = s->tryAllocate(size, false);
void* p = s->tryAllocate(size);
if (p == 0) abort();
return p;
}
@ -187,7 +167,7 @@ class MySystem: public System {
}
virtual void dispose() {
s->free(this, sizeof(*this), false);
s->free(this);
}
pthread_t thread;
@ -215,7 +195,7 @@ class MySystem: public System {
virtual void dispose() {
pthread_mutex_destroy(&mutex);
s->free(this, sizeof(*this), false);
s->free(this);
}
System* s;
@ -414,7 +394,7 @@ class MySystem: public System {
virtual void dispose() {
expect(s, owner_ == 0);
pthread_mutex_destroy(&mutex);
s->free(this, sizeof(*this), false);
s->free(this);
}
System* s;
@ -445,7 +425,7 @@ class MySystem: public System {
int r UNUSED = pthread_key_delete(key);
expect(s, r == 0);
s->free(this, sizeof(*this), false);
s->free(this);
}
System* s;
@ -472,7 +452,7 @@ class MySystem: public System {
if (start_) {
munmap(start_, length_);
}
s->free(this, sizeof(*this), false);
s->free(this);
}
System* s;
@ -525,10 +505,10 @@ class MySystem: public System {
}
if (name_) {
s->free(name_, nameLength + 1, false);
s->free(name_);
}
s->free(this, sizeof(*this), false);
s->free(this);
}
System* s;
@ -540,49 +520,63 @@ class MySystem: public System {
System::Library* next_;
};
MySystem() {
struct sigaction sa;
memset(&sa, 0, sizeof(struct sigaction));
sigemptyset(&(sa.sa_mask));
sa.sa_flags = SA_SIGINFO;
sa.sa_sigaction = handleSignal;
MySystem(): threadVisitor(0), visitTarget(0) {
expect(this, system == 0);
system = this;
registerHandler(&nullHandler, InterruptSignalIndex);
registerHandler(&nullHandler, VisitSignalIndex);
expect(this, make(&visitLock) == 0);
}
int registerHandler(System::SignalHandler* handler, int index) {
if (handler) {
handlers[index] = handler;
struct sigaction sa;
memset(&sa, 0, sizeof(struct sigaction));
sigemptyset(&(sa.sa_mask));
sa.sa_flags = SA_SIGINFO;
sa.sa_sigaction = handleSignal;
int rv UNUSED = sigaction(InterruptSignal, &sa, 0);
expect(this, rv == 0);
}
virtual void* tryAllocate(unsigned size, bool executable) {
assert(this, (not executable) or (size % LikelyPageSizeInBytes == 0));
#ifndef MAP_32BIT
#define MAP_32BIT 0
#endif
if (executable) {
void* p = mmap(0, size, PROT_EXEC | PROT_READ | PROT_WRITE,
MAP_PRIVATE | MAP_ANON | MAP_32BIT, -1, 0);
if (p == MAP_FAILED) {
return 0;
} else {
return p;
}
return sigaction(signals[index], &sa, oldHandlers + index);
} else if (handlers[index]) {
handlers[index] = 0;
return sigaction(signals[index], oldHandlers + index, 0);
} else {
return malloc(size);
return 1;
}
}
virtual void free(const void* p, unsigned size, bool executable) {
if (p) {
if (executable) {
int r UNUSED = munmap(const_cast<void*>(p), size);
assert(this, r == 0);
} else {
::free(const_cast<void*>(p));
}
virtual void* tryAllocate(unsigned sizeInBytes) {
return malloc(sizeInBytes);
}
virtual void free(const void* p) {
if (p) ::free(const_cast<void*>(p));
}
virtual void* tryAllocateExecutable(unsigned sizeInBytes) {
assert(this, sizeInBytes % LikelyPageSizeInBytes == 0);
void* p = mmap(0, sizeInBytes, PROT_EXEC | PROT_READ | PROT_WRITE,
MAP_PRIVATE | MAP_ANON, -1, 0);
if (p == MAP_FAILED) {
return 0;
} else {
return p;
}
}
virtual void freeExecutable(const void* p, unsigned sizeInBytes) {
assert(this, sizeInBytes % LikelyPageSizeInBytes == 0);
int r UNUSED = munmap(const_cast<void*>(p), sizeInBytes);
assert(this, r == 0);
}
virtual bool success(Status s) {
return s == 0;
}
@ -618,22 +612,32 @@ class MySystem: public System {
}
virtual Status handleSegFault(SignalHandler* handler) {
if (handler) {
segFaultHandler = handler;
return registerHandler(handler, SegFaultSignalIndex);
}
struct sigaction sa;
memset(&sa, 0, sizeof(struct sigaction));
sigemptyset(&(sa.sa_mask));
sa.sa_flags = SA_SIGINFO;
sa.sa_sigaction = handleSignal;
return sigaction(SegFaultSignal, &sa, &oldSegFaultHandler);
} else if (segFaultHandler) {
segFaultHandler = 0;
return sigaction(SegFaultSignal, &oldSegFaultHandler, 0);
} else {
return 1;
}
virtual Status visit(System::Thread* st, System::Thread* sTarget,
ThreadVisitor* visitor)
{
assert(this, st != sTarget);
Thread* t = static_cast<Thread*>(st);
Thread* target = static_cast<Thread*>(sTarget);
ACQUIRE_MONITOR(t, visitLock);
while (threadVisitor) visitLock->wait(t, 0);
threadVisitor = visitor;
visitTarget = target;
int rv = pthread_kill(target->thread, VisitSignal);
expect(this, rv == 0);
while (visitTarget) visitLock->wait(t, 0);
threadVisitor = 0;
return 0;
}
virtual uint64_t call(void* function, uintptr_t* arguments, uint8_t* types,
@ -687,18 +691,18 @@ class MySystem: public System {
bool alreadyAllocated = false;
bool isMain = false;
unsigned nameLength = (name ? strlen(name) : 0);
if (mapName) {
if (mapName and name) {
unsigned size = nameLength + 3 + sizeof(SO_SUFFIX);
char buffer[size];
snprintf(buffer, size, "lib%s" SO_SUFFIX, name);
p = dlopen(buffer, RTLD_LAZY);
p = dlopen(buffer, RTLD_LAZY | RTLD_LOCAL);
} else {
if (!name) {
pathOfExecutable(this, &name, &nameLength);
alreadyAllocated = true;
isMain = true;
}
p = dlopen(name, RTLD_LAZY);
p = dlopen(name, RTLD_LAZY | RTLD_LOCAL);
}
if (p) {
@ -711,7 +715,7 @@ class MySystem: public System {
n = static_cast<char*>(allocate(this, nameLength + 1));
memcpy(n, name, nameLength + 1);
if (alreadyAllocated) {
free(name, nameLength, false);
free(name);
}
} else {
n = 0;
@ -747,10 +751,89 @@ class MySystem: public System {
}
virtual void dispose() {
visitLock->dispose();
registerHandler(0, InterruptSignalIndex);
registerHandler(0, VisitSignalIndex);
system = 0;
::free(this);
}
class NullSignalHandler: public SignalHandler {
virtual bool handleSignal(void**, void**, void**, void**) { return false; }
} nullHandler;
SignalHandler* handlers[3];
struct sigaction oldHandlers[3];
ThreadVisitor* threadVisitor;
Thread* visitTarget;
System::Monitor* visitLock;
};
void
handleSignal(int signal, siginfo_t* info, void* context)
{
ucontext_t* c = static_cast<ucontext_t*>(context);
void* ip = reinterpret_cast<void*>(IP_REGISTER(c));
void* base = reinterpret_cast<void*>(BASE_REGISTER(c));
void* stack = reinterpret_cast<void*>(STACK_REGISTER(c));
void* thread = reinterpret_cast<void*>(THREAD_REGISTER(c));
unsigned index;
switch (signal) {
case VisitSignal: {
index = VisitSignalIndex;
system->threadVisitor->visit(ip, base, stack);
System::Thread* t = system->visitTarget;
system->visitTarget = 0;
system->visitLock->notifyAll(t);
} break;
case SegFaultSignal: {
index = SegFaultSignalIndex;
bool jump = system->handlers[index]->handleSignal
(&ip, &base, &stack, &thread);
if (jump) {
// I'd like to use setcontext here (and get rid of the
// sigprocmask call), but it doesn't work on my Linux x86_64
// system, and I can't tell from the documentation if it's even
// supposed to work.
sigset_t set;
sigemptyset(&set);
sigaddset(&set, SegFaultSignal);
sigprocmask(SIG_UNBLOCK, &set, 0);
vmJump(ip, base, stack, thread);
}
} break;
case InterruptSignal: {
index = InterruptSignalIndex;
} break;
default: abort();
}
if (system->oldHandlers[index].sa_flags & SA_SIGINFO) {
system->oldHandlers[index].sa_sigaction(signal, info, context);
} else if (system->oldHandlers[index].sa_handler) {
system->oldHandlers[index].sa_handler(signal);
} else {
abort();
}
}
} // namespace
namespace vm {

View File

@ -191,50 +191,6 @@ resolveNativeMethod2(Thread* t, object method)
return 0;
}
ExceptionHandler*
findExceptionHandler(Thread* t, object method, unsigned ip)
{
PROTECT(t, method);
object eht = codeExceptionHandlerTable(t, methodCode(t, method));
if (eht) {
for (unsigned i = 0; i < exceptionHandlerTableLength(t, eht); ++i) {
ExceptionHandler* eh = exceptionHandlerTableBody(t, eht, i);
if (ip - 1 >= exceptionHandlerStart(eh)
and ip - 1 < exceptionHandlerEnd(eh))
{
object catchType = 0;
if (exceptionHandlerCatchType(eh)) {
object e = t->exception;
t->exception = 0;
PROTECT(t, e);
PROTECT(t, eht);
catchType = resolveClassInPool
(t, codePool(t, methodCode(t, method)),
exceptionHandlerCatchType(eh) - 1);
if (catchType) {
eh = exceptionHandlerTableBody(t, eht, i);
t->exception = e;
} else {
// can't find what we're supposed to catch - move on.
continue;
}
}
if (catchType == 0 or instanceOf(t, catchType, t->exception)) {
return eh;
}
}
}
}
return 0;
}
int
findLineNumber(Thread* t, object method, unsigned ip)
{

View File

@ -40,8 +40,7 @@ inline object
resolveClassInObject(Thread* t, object container, unsigned classOffset)
{
object o = cast<object>(container, classOffset);
if (objectClass(t, o) == arrayBody(t, t->m->types, Machine::ByteArrayType))
{
if (objectClass(t, o) == arrayBody(t, t->m->types, Machine::ByteArrayType)) {
PROTECT(t, container);
o = resolveClass(t, o);
@ -56,8 +55,7 @@ inline object
resolveClassInPool(Thread* t, object pool, unsigned index)
{
object o = singletonObject(t, pool, index);
if (objectClass(t, o) == arrayBody(t, t->m->types, Machine::ByteArrayType))
{
if (objectClass(t, o) == arrayBody(t, t->m->types, Machine::ByteArrayType)) {
PROTECT(t, pool);
o = resolveClass(t, o);
@ -194,9 +192,6 @@ populateMultiArray(Thread* t, object array, int32_t* counts,
}
}
ExceptionHandler*
findExceptionHandler(Thread* t, object method, unsigned ip);
int
findLineNumber(Thread* t, object method, unsigned ip);

View File

@ -115,6 +115,9 @@ class Processor {
virtual void
dispose() = 0;
virtual object
getStackTrace(Thread* t, Thread* target) = 0;
object
invoke(Thread* t, object method, object this_, ...)
{

View File

@ -34,6 +34,12 @@ class System {
virtual void dispose() = 0;
};
class ThreadVisitor {
public:
virtual ~ThreadVisitor() { }
virtual void visit(void* ip, void* base, void* stack) = 0;
};
class Runnable {
public:
virtual ~Runnable() { }
@ -99,17 +105,36 @@ class System {
void** thread) = 0;
};
class MonitorResource {
public:
MonitorResource(System::Thread* t, System::Monitor* m): t(t), m(m) {
m->acquire(t);
}
~MonitorResource() {
m->release(t);
}
private:
System::Thread* t;
System::Monitor* m;
};
virtual ~System() { }
virtual bool success(Status) = 0;
virtual void* tryAllocate(unsigned size, bool executable) = 0;
virtual void free(const void* p, unsigned size, bool executable) = 0;
virtual void* tryAllocate(unsigned sizeInBytes) = 0;
virtual void free(const void* p) = 0;
virtual void* tryAllocateExecutable(unsigned sizeInBytes) = 0;
virtual void freeExecutable(const void* p, unsigned sizeInBytes) = 0;
virtual Status attach(Runnable*) = 0;
virtual Status start(Runnable*) = 0;
virtual Status make(Mutex**) = 0;
virtual Status make(Monitor**) = 0;
virtual Status make(Local**) = 0;
virtual Status handleSegFault(SignalHandler* handler) = 0;
virtual Status visit(Thread* thread, Thread* target,
ThreadVisitor* visitor) = 0;
virtual uint64_t call(void* function, uintptr_t* arguments, uint8_t* types,
unsigned count, unsigned size,
unsigned returnType) = 0;
@ -123,6 +148,9 @@ class System {
virtual void dispose() = 0;
};
#define ACQUIRE_MONITOR(t, m) \
System::MonitorResource MAKE_NAME(monitorResource_) (t, m)
inline void NO_RETURN
abort(System* s)
{

View File

@ -1507,7 +1507,7 @@ writeOffset(Output* out, Object* offset, bool allocationStyle = false)
out->write("length");
} else {
out->write(typeName(memberOwner(o)));
out->write(capitalize("length"));
out->write("Length");
out->write("(o)");
}
out->write(" * ");
@ -1762,6 +1762,49 @@ typeFixedSize(Object* type)
return length;
}
unsigned
typeArrayElementSize(Object* type)
{
for (MemberIterator it(type); it.hasMore();) {
Object* m = it.next();
switch (m->type) {
case Object::Scalar: break;
case Object::Array: {
return memberElementSize(m);
} break;
default: UNREACHABLE;
}
}
return 0;
}
void
writeSizes(Output* out, Object* declarations)
{
for (Object* p = declarations; p; p = cdr(p)) {
Object* o = car(p);
switch (o->type) {
case Object::Type: {
out->write("const unsigned FixedSizeOf");
out->write(capitalize(typeName(o)));
out->write(" = ");
out->write(typeFixedSize(o));
out->write(";\n\n");
out->write("const unsigned ArrayElementSizeOf");
out->write(capitalize(typeName(o)));
out->write(" = ");
out->write(typeArrayElementSize(o));
out->write(";\n\n");
} break;
default: break;
}
}
}
const char*
obfuscate(const char* s)
{
@ -2049,24 +2092,6 @@ set(uint32_t* mask, unsigned index)
}
}
unsigned
typeArrayElementSize(Object* type)
{
for (MemberIterator it(type); it.hasMore();) {
Object* m = it.next();
switch (m->type) {
case Object::Scalar: break;
case Object::Array: {
return memberElementSize(m);
} break;
default: UNREACHABLE;
}
}
return 0;
}
uint32_t
typeObjectMask(Object* type)
{
@ -2274,6 +2299,7 @@ main(int ac, char** av)
writePods(&out, declarations);
writeAccessors(&out, declarations);
writeSizes(&out, declarations);
writeInitializerDeclarations(&out, declarations);
writeConstructorDeclarations(&out, declarations);
}

View File

@ -24,10 +24,10 @@
(void* value))
(pod exceptionHandler
(uint32_t start)
(uint32_t end)
(uint32_t ip)
(uint32_t catchType))
(uint16_t start)
(uint16_t end)
(uint16_t ip)
(uint16_t catchType))
(type exceptionHandlerTable
(array exceptionHandler body))
@ -91,13 +91,22 @@
(object method)
(int ip))
(type traceNode
(type treeNode
(object value)
(object left)
(object right))
(type treePath
(uintptr_t fresh)
(object node)
(object root)
(object ancestors))
(type callNode
(intptr_t address)
(object next)
(object method)
(object target)
(uintptr_t virtualCall)
(array uintptr_t map))
(object next))
(type array
(noassert array object body))

View File

@ -14,18 +14,241 @@ using namespace vm;
namespace {
object
clone(Thread* t, object o)
inline object
getTreeNodeValue(Thread*, object n)
{
object class_ = objectClass(t, o);
unsigned size = baseSize(t, o, class_) * BytesPerWord;
return reinterpret_cast<object>
(cast<intptr_t>(n, TreeNodeValue) & PointerMask);
}
object clone = make(t, class_);
memcpy(reinterpret_cast<void**>(clone) + 1,
reinterpret_cast<void**>(o) + 1,
size - BytesPerWord);
inline void
setTreeNodeValue(Thread* t, object n, object value)
{
intptr_t red = cast<intptr_t>(n, TreeNodeValue) & (~PointerMask);
return clone;
set(t, n, TreeNodeValue, value);
cast<intptr_t>(n, TreeNodeValue) |= red;
}
inline bool
treeNodeRed(Thread*, object n)
{
return (cast<intptr_t>(n, TreeNodeValue) & (~PointerMask)) == 1;
}
inline void
setTreeNodeRed(Thread*, object n, bool red)
{
if (red) {
cast<intptr_t>(n, TreeNodeValue) |= 1;
} else {
cast<intptr_t>(n, TreeNodeValue) &= PointerMask;
}
}
inline object
cloneTreeNode(Thread* t, object n)
{
PROTECT(t, n);
object newNode = makeTreeNode
(t, getTreeNodeValue(t, n), treeNodeLeft(t, n), treeNodeRight(t, n));
setTreeNodeRed(t, newNode, treeNodeRed(t, n));
return newNode;
}
object
treeFind(Thread* t, object old, object node, object sentinal,
intptr_t (*compare)(Thread* t, object a, object b))
{
PROTECT(t, old);
PROTECT(t, node);
PROTECT(t, sentinal);
object newRoot = cloneTreeNode(t, old);
PROTECT(t, newRoot);
object new_ = newRoot;
PROTECT(t, new_);
object ancestors = 0;
PROTECT(t, ancestors);
while (old != sentinal) {
ancestors = makePair(t, new_, ancestors);
intptr_t difference = compare
(t, getTreeNodeValue(t, node), getTreeNodeValue(t, old));
if (difference < 0) {
old = treeNodeLeft(t, old);
object n = cloneTreeNode(t, old);
set(t, new_, TreeNodeLeft, n);
new_ = n;
} else if (difference > 0) {
old = treeNodeRight(t, old);
object n = cloneTreeNode(t, old);
set(t, new_, TreeNodeRight, n);
new_ = n;
} else {
return makeTreePath(t, false, new_, newRoot, pairSecond(t, ancestors));
}
}
setTreeNodeValue(t, new_, getTreeNodeValue(t, node));
return makeTreePath(t, true, new_, newRoot, ancestors);
}
object
leftRotate(Thread* t, object n)
{
PROTECT(t, n);
object child = cloneTreeNode(t, treeNodeRight(t, n));
set(t, n, TreeNodeRight, treeNodeLeft(t, child));
set(t, child, TreeNodeLeft, n);
return child;
}
object
rightRotate(Thread* t, object n)
{
PROTECT(t, n);
object child = cloneTreeNode(t, treeNodeLeft(t, n));
set(t, n, TreeNodeLeft, treeNodeRight(t, child));
set(t, child, TreeNodeRight, n);
return child;
}
object
treeAdd(Thread* t, object path)
{
object new_ = treePathNode(t, path);
PROTECT(t, new_);
object newRoot = treePathRoot(t, path);
PROTECT(t, newRoot);
object ancestors = treePathAncestors(t, path);
PROTECT(t, ancestors);
// rebalance
setTreeNodeRed(t, new_, true);
while (ancestors != 0 and treeNodeRed(t, pairFirst(t, ancestors))) {
if (pairFirst(t, ancestors)
== treeNodeLeft(t, pairFirst(t, pairSecond(t, ancestors))))
{
if (treeNodeRed
(t, treeNodeRight(t, pairFirst(t, pairSecond(t, ancestors)))))
{
setTreeNodeRed(t, pairFirst(t, ancestors), true);
object n = cloneTreeNode
(t, treeNodeRight(t, pairFirst(t, pairSecond(t, ancestors))));
set(t, pairFirst(t, pairSecond(t, ancestors)), TreeNodeRight, n);
setTreeNodeRed
(t, treeNodeRight
(t, pairFirst(t, pairSecond(t, ancestors))), false);
setTreeNodeRed(t, pairFirst(t, pairSecond(t, ancestors)), false);
new_ = pairFirst(t, pairSecond(t, ancestors));
ancestors = pairSecond(t, pairSecond(t, ancestors));
} else {
if (new_ == treeNodeRight(t, pairFirst(t, ancestors))) {
new_ = pairFirst(t, ancestors);
ancestors = pairSecond(t, ancestors);
object n = leftRotate(t, new_);
if (new_ == treeNodeRight(t, pairFirst(t, ancestors))) {
set(t, pairFirst(t, ancestors), TreeNodeRight, n);
} else {
set(t, pairFirst(t, ancestors), TreeNodeLeft, n);
}
ancestors = makePair(t, n, ancestors);
}
setTreeNodeRed(t, pairFirst(t, ancestors), false);
setTreeNodeRed(t, pairFirst(t, pairSecond(t, ancestors)), true);
object n = rightRotate(t, pairFirst(t, pairSecond(t, ancestors)));
if (pairSecond(t, pairSecond(t, ancestors)) == 0) {
newRoot = n;
} else if (treeNodeRight
(t, pairFirst(t, pairSecond(t, pairSecond(t, ancestors))))
== pairFirst(t, pairSecond(t, ancestors)))
{
set(t, pairFirst(t, pairSecond(t, pairSecond(t, ancestors))),
TreeNodeRight, n);
} else {
set(t, pairFirst(t, pairSecond(t, pairSecond(t, ancestors))),
TreeNodeLeft, n);
}
// done
}
} else { // this is just the reverse of the code above (right and
// left swapped):
if (treeNodeRed
(t, treeNodeLeft(t, pairFirst(t, pairSecond(t, ancestors)))))
{
setTreeNodeRed(t, pairFirst(t, ancestors), true);
object n = cloneTreeNode
(t, treeNodeLeft(t, pairFirst(t, pairSecond(t, ancestors))));
set(t, pairFirst(t, pairSecond(t, ancestors)), TreeNodeLeft, n);
setTreeNodeRed
(t, treeNodeLeft
(t, pairFirst(t, pairSecond(t, ancestors))), false);
setTreeNodeRed(t, pairFirst(t, pairSecond(t, ancestors)), false);
new_ = pairFirst(t, pairSecond(t, ancestors));
ancestors = pairSecond(t, pairSecond(t, ancestors));
} else {
if (new_ == treeNodeLeft(t, pairFirst(t, ancestors))) {
new_ = pairFirst(t, ancestors);
ancestors = pairSecond(t, ancestors);
object n = rightRotate(t, new_);
if (new_ == treeNodeLeft(t, pairFirst(t, ancestors))) {
set(t, pairFirst(t, ancestors), TreeNodeLeft, n);
} else {
set(t, pairFirst(t, ancestors), TreeNodeRight, n);
}
ancestors = makePair(t, n, ancestors);
}
setTreeNodeRed(t, pairFirst(t, ancestors), false);
setTreeNodeRed(t, pairFirst(t, pairSecond(t, ancestors)), true);
object n = leftRotate(t, pairFirst(t, pairSecond(t, ancestors)));
if (pairSecond(t, pairSecond(t, ancestors)) == 0) {
newRoot = n;
} else if (treeNodeLeft
(t, pairFirst(t, pairSecond(t, pairSecond(t, ancestors))))
== pairFirst(t, pairSecond(t, ancestors)))
{
set(t, pairFirst(t, pairSecond(t, pairSecond(t, ancestors))),
TreeNodeLeft, n);
} else {
set(t, pairFirst(t, pairSecond(t, pairSecond(t, ancestors))),
TreeNodeRight, n);
}
// done
}
}
}
setTreeNodeRed(t, newRoot, false);
return newRoot;
}
} // namespace
@ -293,4 +516,40 @@ vectorAppend(Thread* t, object vector, object value)
return vector;
}
object
treeQuery(Thread* t, object tree, intptr_t key, object sentinal,
intptr_t (*compare)(Thread* t, intptr_t key, object b))
{
object node = tree;
while (node != sentinal) {
intptr_t difference = compare(t, key, getTreeNodeValue(t, node));
if (difference < 0) {
node = treeNodeLeft(t, node);
} else if (difference > 0) {
node = treeNodeRight(t, node);
} else {
return getTreeNodeValue(t, node);
}
}
return 0;
}
object
treeInsert(Thread* t, object tree, object value, object sentinal,
intptr_t (*compare)(Thread* t, object a, object b))
{
PROTECT(t, tree);
PROTECT(t, sentinal);
object node = makeTreeNode(t, value, sentinal, sentinal);
object path = treeFind(t, tree, node, sentinal, compare);
if (treePathFresh(t, path)) {
return treeAdd(t, path);
} else {
return tree;
}
}
} // namespace vm

View File

@ -83,6 +83,14 @@ listAppend(Thread* t, object list, object value);
object
vectorAppend(Thread* t, object vector, object value);
object
treeQuery(Thread* t, object tree, intptr_t key, object sentinal,
intptr_t (*compare)(Thread* t, intptr_t key, object b));
object
treeInsert(Thread* t, object tree, object value, object sentinal,
intptr_t (*compare)(Thread* t, object a, object b));
} // vm
#endif//UTIL_H

View File

@ -32,7 +32,7 @@ class Vector {
void dispose() {
if (data and minimumCapacity >= 0) {
allocator->free(data, capacity, false);
allocator->free(data, capacity);
}
}
@ -52,10 +52,10 @@ class Vector {
unsigned newCapacity = max
(position + space, max(minimumCapacity, capacity * 2));
uint8_t* newData = static_cast<uint8_t*>
(allocator->allocate(newCapacity, false));
(allocator->allocate(newCapacity));
if (data) {
memcpy(newData, data, position);
allocator->free(data, capacity, false);
allocator->free(data, capacity);
}
data = newData;
capacity = newCapacity;

View File

@ -23,6 +23,23 @@ using namespace vm;
namespace {
class MutexResource {
public:
MutexResource(System* s, HANDLE m): s(s), m(m) {
int r UNUSED = WaitForSingleObject(m, INFINITE);
assert(s, r == WAIT_OBJECT_0);
}
~MutexResource() {
bool success UNUSED = ReleaseMutex(m);
assert(s, success);
}
private:
System* s;
HANDLE m;
};
System::SignalHandler* segFaultHandler = 0;
LPTOP_LEVEL_EXCEPTION_FILTER oldSegFaultHandler = 0;
@ -43,23 +60,6 @@ handleException(LPEXCEPTION_POINTERS e)
return EXCEPTION_CONTINUE_SEARCH;
}
class MutexResource {
public:
MutexResource(System* s, HANDLE m): s(s), m(m) {
int r UNUSED = WaitForSingleObject(m, INFINITE);
assert(s, r == WAIT_OBJECT_0);
}
~MutexResource() {
bool success UNUSED = ReleaseMutex(m);
assert(s, success);
}
private:
System* s;
HANDLE m;
};
DWORD WINAPI
run(void* r)
{
@ -70,7 +70,7 @@ run(void* r)
void*
allocate(System* s, unsigned size)
{
void* p = s->tryAllocate(size, false);
void* p = s->tryAllocate(size);
if (p == 0) abort();
return p;
}
@ -117,7 +117,7 @@ class MySystem: public System {
CloseHandle(event);
CloseHandle(mutex);
CloseHandle(thread);
s->free(this, sizeof(*this), false);
s->free(this);
}
HANDLE thread;
@ -148,7 +148,7 @@ class MySystem: public System {
virtual void dispose() {
CloseHandle(mutex);
s->free(this, sizeof(*this), false);
s->free(this);
}
System* s;
@ -362,7 +362,7 @@ class MySystem: public System {
virtual void dispose() {
assert(s, owner_ == 0);
CloseHandle(mutex);
s->free(this, sizeof(*this), false);
s->free(this);
}
System* s;
@ -393,7 +393,7 @@ class MySystem: public System {
bool r UNUSED = TlsFree(key);
assert(s, r);
s->free(this, sizeof(*this), false);
s->free(this);
}
System* s;
@ -425,7 +425,7 @@ class MySystem: public System {
if (mapping) CloseHandle(mapping);
if (file) CloseHandle(file);
}
system->free(this, sizeof(*this), false);
system->free(this);
}
System* system;
@ -437,12 +437,10 @@ class MySystem: public System {
class Library: public System::Library {
public:
Library(System* s, HMODULE handle, const char* name, size_t nameLength,
bool mapName):
Library(System* s, HMODULE handle, const char* name, bool mapName):
s(s),
handle(handle),
name_(name),
nameLength(nameLength),
mapName_(mapName),
next_(0)
{ }
@ -484,16 +482,15 @@ class MySystem: public System {
}
if (name_) {
s->free(name_, nameLength + 1, false);
s->free(name_);
}
s->free(this, sizeof(*this), false);
s->free(this);
}
System* s;
HMODULE handle;
const char* name_;
size_t nameLength;
bool mapName_;
System::Library* next_;
};
@ -503,14 +500,26 @@ class MySystem: public System {
assert(this, mutex);
}
virtual void* tryAllocate(unsigned size, bool) {
return malloc(size);
virtual void* tryAllocate(unsigned sizeInBytes) {
return malloc(sizeInBytes);
}
virtual void free(const void* p, unsigned, bool) {
virtual void free(const void* p) {
if (p) ::free(const_cast<void*>(p));
}
virtual void* tryAllocateExecutable(unsigned sizeInBytes) {
assert(this, sizeInBytes % LikelyPageSizeInBytes == 0);
return VirtualAlloc
(0, sizeInBytes, MEM_COMMIT | MEM_RESERVE, PAGE_EXECUTE_READWRITE);
}
virtual void freeExecutable(const void* p, unsigned) {
int r UNUSED = VirtualFree(const_cast<void*>(p), 0, MEM_RELEASE);
assert(this, r);
}
virtual bool success(Status s) {
return s == 0;
}
@ -564,6 +573,32 @@ class MySystem: public System {
}
}
virtual Status visit(System::Thread* st UNUSED, System::Thread* sTarget,
ThreadVisitor* visitor)
{
assert(this, st != sTarget);
Thread* target = static_cast<Thread*>(sTarget);
ACQUIRE(this, mutex);
int rv = SuspendThread(target->thread);
expect(this, rv != -1);
CONTEXT context;
rv = GetThreadContext(target->thread, &context);
expect(this, rv);
visitor->visit(reinterpret_cast<void*>(context.Eip),
reinterpret_cast<void*>(context.Ebp),
reinterpret_cast<void*>(context.Esp));
rv = ResumeThread(target->thread);
expect(this, rv != -1);
return 0;
}
virtual uint64_t call(void* function, uintptr_t* arguments, uint8_t* types,
unsigned count, unsigned size, unsigned returnType)
{
@ -623,7 +658,7 @@ class MySystem: public System {
{
HMODULE handle;
unsigned nameLength = (name ? strlen(name) : 0);
if (mapName) {
if (mapName and name) {
unsigned size = sizeof(SO_PREFIX) + nameLength + sizeof(SO_SUFFIX);
char buffer[size];
snprintf(buffer, size, SO_PREFIX "%s" SO_SUFFIX, name);
@ -648,7 +683,7 @@ class MySystem: public System {
}
*lib = new (allocate(this, sizeof(Library)))
Library(this, handle, n, nameLength, mapName);
Library(this, handle, n, mapName);
return 0;
} else {

View File

@ -214,21 +214,22 @@ encode(Context* c, uint8_t* instruction, unsigned length, int a, int b,
}
void
rex(Context* c)
rex(Context* c, uint8_t mask, int r)
{
if (BytesPerWord == 8) {
c->code.append(0x48);
c->code.append(mask | ((r & 8) >> 3));
}
}
void
rex(Context* c)
{
rex(c, 0x48, rax);
}
void
encode(Context* c, uint8_t instruction, int a, Assembler::Memory* b, bool rex)
{
if (b->traceHandler) {
fprintf(stderr, "handle trace %p\n", b->traceHandler);
b->traceHandler->handleTrace(codePromise(c, c->code.length()));
}
if (rex) {
::rex(c);
}
@ -240,11 +241,6 @@ void
encode2(Context* c, uint16_t instruction, int a, Assembler::Memory* b,
bool rex)
{
if (b->traceHandler) {
fprintf(stderr, "handle trace %p\n", b->traceHandler);
b->traceHandler->handleTrace(codePromise(c, c->code.length()));
}
if (rex) {
::rex(c);
}
@ -292,18 +288,31 @@ conditional(Context* c, unsigned condition, Assembler::Constant* a)
}
void
callC(Context* c, unsigned size UNUSED, Assembler::Constant* a)
callC(Context* c, unsigned size, Assembler::Constant* a)
{
assert(c, size == BytesPerWord);
unconditional(c, 0xe8, a);
if (BytesPerWord == 8) {
Assembler::Register r(r10);
moveCR(c, size, a, &r);
callR(c, size, &r);
c->client->releaseTemporary(r.low);
} else {
unconditional(c, 0xe8, a);
}
}
void
alignedCallC(Context* c, unsigned size, Assembler::Constant* a)
{
while ((c->code.length() + 1) % 4) {
c->code.append(0x90);
if (BytesPerWord == 8) {
while ((c->code.length() + 2) % 8) {
c->code.append(0x90);
}
} else {
while ((c->code.length() + 1) % 4) {
c->code.append(0x90);
}
}
callC(c, size, a);
}
@ -313,6 +322,7 @@ callR(Context* c, unsigned size UNUSED, Assembler::Register* a)
{
assert(c, size == BytesPerWord);
if (a->low & 8) rex(c, 0x40, a->low);
c->code.append(0xff);
c->code.append(0xd0 | a->low);
}
@ -521,7 +531,7 @@ moveCR(Context* c, unsigned size UNUSED, Assembler::Constant* a,
{
assert(c, BytesPerWord == 8 or size == 4); // todo
rex(c);
rex(c, 0x48, b->low);
c->code.append(0xb8 | b->low);
if (a->value->resolved()) {
c->code.appendAddress(a->value->value());
@ -1131,13 +1141,26 @@ class MyAssembler: public Assembler {
}
virtual void updateCall(void* returnAddress, void* newTarget) {
uint8_t* instruction = static_cast<uint8_t*>(returnAddress) - 5;
assert(&c, *instruction == 0xE8);
assert(&c, reinterpret_cast<uintptr_t>(instruction + 1) % 4 == 0);
if (BytesPerWord == 8) {
uint8_t* instruction = static_cast<uint8_t*>(returnAddress) - 13;
assert(&c, instruction[0] == 0x49);
assert(&c, instruction[1] == 0xba);
assert(&c, instruction[10] == 0x41);
assert(&c, instruction[11] == 0xff);
assert(&c, instruction[12] == 0xd2);
assert(&c, reinterpret_cast<uintptr_t>(instruction + 2) % 8 == 0);
int32_t v = static_cast<uint8_t*>(newTarget)
- static_cast<uint8_t*>(returnAddress);
memcpy(instruction + 1, &v, 4);
intptr_t v = reinterpret_cast<intptr_t>(newTarget);
memcpy(instruction + 2, &v, 8);
} else {
uint8_t* instruction = static_cast<uint8_t*>(returnAddress) - 5;
assert(&c, *instruction == 0xE8);
assert(&c, reinterpret_cast<uintptr_t>(instruction + 1) % 4 == 0);
int32_t v = static_cast<uint8_t*>(newTarget)
- static_cast<uint8_t*>(returnAddress);
memcpy(instruction + 1, &v, 4);
}
}
virtual void dispose() {

View File

@ -27,11 +27,9 @@ class Zone: public Allocator {
uint8_t data[0];
};
Zone(System* s, Allocator* allocator, bool executable,
unsigned minimumFootprint):
Zone(System* s, Allocator* allocator, unsigned minimumFootprint):
s(s),
allocator(allocator),
executable(executable),
segment(0),
position(0),
minimumFootprint(minimumFootprint < sizeof(Segment) ? 0 :
@ -45,11 +43,11 @@ class Zone: public Allocator {
void dispose() {
for (Segment* seg = segment, *next; seg; seg = next) {
next = seg->next;
allocator->free(seg, sizeof(Segment) + seg->size, executable);
allocator->free(seg, sizeof(Segment) + seg->size);
}
}
bool ensure(unsigned space, bool executable) {
bool ensure(unsigned space) {
if (segment == 0 or position + space > segment->size) {
unsigned size = max
(space, max
@ -60,10 +58,10 @@ class Zone: public Allocator {
size = (size + (LikelyPageSizeInBytes - 1))
& ~(LikelyPageSizeInBytes - 1);
void* p = allocator->tryAllocate(size, executable);
void* p = allocator->tryAllocate(size);
if (p == 0) {
size = space + sizeof(Segment);
void* p = allocator->tryAllocate(size, executable);
void* p = allocator->tryAllocate(size);
if (p == 0) {
return false;
}
@ -75,11 +73,9 @@ class Zone: public Allocator {
return true;
}
virtual void* tryAllocate(unsigned size, bool executable) {
assert(s, executable == this->executable);
virtual void* tryAllocate(unsigned size) {
size = pad(size);
if (ensure(size, executable)) {
if (ensure(size)) {
void* r = segment->data + position;
position += size;
return r;
@ -88,27 +84,20 @@ class Zone: public Allocator {
}
}
virtual void* allocate(unsigned size, bool executable) {
assert(s, executable == this->executable);
void* p = tryAllocate(size, executable);
virtual void* allocate(unsigned size) {
void* p = tryAllocate(size);
expect(s, p);
return p;
}
virtual void free(const void*, unsigned, bool) {
virtual void free(const void*, unsigned) {
// not supported
abort(s);
}
void* allocate(unsigned size) {
return allocate(size, executable);
}
System* s;
Allocator* allocator;
void* context;
bool executable;
Segment* segment;
unsigned position;
unsigned minimumFootprint;

View File

@ -2,7 +2,22 @@ public class Misc {
private static int alpha;
private static int beta;
private static byte byte1, byte2, byte3;
private int gamma;
private int pajama;
private boolean boolean1;
private boolean boolean2;
private long time;
public Misc() {
expect(! boolean1);
expect(! boolean2);
time = 0xffffffffffffffffL;
expect(! boolean1);
expect(! boolean2);
}
private String foo(String s) {
return s;