mirror of
https://github.com/corda/corda.git
synced 2025-01-19 11:16:54 +00:00
Merge branch 'master' into compiler
Conflicts: src/compile.cpp src/compiler.cpp src/compiler.h
This commit is contained in:
commit
525f733171
@ -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)
|
||||
{
|
||||
|
@ -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;
|
||||
}
|
||||
|
||||
|
@ -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;
|
||||
|
@ -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 { }
|
||||
}
|
||||
|
21
classpath/java/lang/ArithmeticException.java
Normal file
21
classpath/java/lang/ArithmeticException.java
Normal 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);
|
||||
}
|
||||
}
|
@ -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');
|
||||
}
|
||||
|
@ -376,6 +376,10 @@ public final class Class <T> {
|
||||
return flags;
|
||||
}
|
||||
|
||||
public boolean isInterface() {
|
||||
return (flags & Modifier.INTERFACE) != 0;
|
||||
}
|
||||
|
||||
public Class getSuperclass() {
|
||||
return super_;
|
||||
}
|
||||
|
@ -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;
|
||||
}
|
||||
|
@ -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) {
|
||||
|
@ -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);
|
||||
|
@ -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);
|
||||
}
|
||||
|
@ -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[])) {
|
||||
|
@ -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; }
|
||||
}
|
||||
|
@ -10,7 +10,7 @@
|
||||
|
||||
package java.net;
|
||||
|
||||
public class InetSocketAddress {
|
||||
public class InetSocketAddress extends SocketAddress {
|
||||
private final String host;
|
||||
private final int port;
|
||||
|
||||
|
17
classpath/java/net/ServerSocket.java
Normal file
17
classpath/java/net/ServerSocket.java
Normal 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;
|
||||
}
|
13
classpath/java/net/SocketAddress.java
Normal file
13
classpath/java/net/SocketAddress.java
Normal 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 { }
|
59
classpath/java/nio/Buffer.java
Normal file
59
classpath/java/nio/Buffer.java
Normal 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;
|
||||
}
|
||||
}
|
@ -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;
|
||||
|
@ -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;
|
||||
}
|
||||
}
|
||||
|
@ -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() {
|
||||
|
@ -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();
|
||||
}
|
||||
|
@ -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;
|
||||
}
|
||||
|
@ -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) {
|
||||
|
@ -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() {
|
||||
|
@ -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);
|
||||
}
|
||||
}
|
@ -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);
|
||||
}
|
||||
}
|
||||
|
@ -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);
|
||||
}
|
||||
|
85
makefile
85
makefile
@ -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 $(@)
|
||||
|
@ -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);
|
||||
}
|
||||
|
||||
|
@ -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
|
||||
|
@ -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 {
|
||||
|
@ -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
29
src/boot.cpp
Normal 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);
|
||||
}
|
||||
|
||||
}
|
@ -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)
|
||||
|
913
src/compile.cpp
913
src/compile.cpp
File diff suppressed because it is too large
Load Diff
@ -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);
|
||||
|
@ -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;
|
||||
|
@ -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(®ion, 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
|
||||
|
@ -27,7 +27,7 @@ class Finder {
|
||||
};
|
||||
|
||||
Finder*
|
||||
makeFinder(System* s, const char* path);
|
||||
makeFinder(System* s, const char* path, const char* bootLibrary);
|
||||
|
||||
} // namespace vm
|
||||
|
||||
|
51
src/heap.cpp
51
src/heap.cpp
@ -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
|
||||
|
@ -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;
|
||||
|
@ -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);
|
||||
}
|
||||
|
188
src/jnienv.cpp
188
src/jnienv.cpp
@ -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);
|
||||
|
121
src/machine.cpp
121
src/machine.cpp
@ -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);
|
||||
}
|
||||
|
@ -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;
|
||||
}
|
||||
|
40
src/main.cpp
40
src/main.cpp
@ -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) {
|
||||
|
293
src/posix.cpp
293
src/posix.cpp
@ -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 {
|
||||
|
@ -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)
|
||||
{
|
||||
|
@ -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);
|
||||
|
||||
|
@ -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_, ...)
|
||||
{
|
||||
|
32
src/system.h
32
src/system.h
@ -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)
|
||||
{
|
||||
|
@ -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);
|
||||
}
|
||||
|
@ -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))
|
||||
|
277
src/util.cpp
277
src/util.cpp
@ -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
|
||||
|
@ -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
|
||||
|
@ -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;
|
||||
|
103
src/windows.cpp
103
src/windows.cpp
@ -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 {
|
||||
|
69
src/x86.cpp
69
src/x86.cpp
@ -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() {
|
||||
|
31
src/zone.h
31
src/zone.h
@ -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;
|
||||
|
@ -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;
|
||||
|
Loading…
Reference in New Issue
Block a user