Skip to content

Commit 87b54cd

Browse files
committedSep 15, 2020
Merge branch 'fibers' of https://github.com/openjdk/loom into fibers
2 parents 0660b7f + aae2af1 commit 87b54cd

File tree

11 files changed

+173
-2272
lines changed

11 files changed

+173
-2272
lines changed
 

‎src/hotspot/share/runtime/vmOperations.cpp

+1-1
Original file line numberDiff line numberDiff line change
@@ -340,7 +340,7 @@ void VM_ThreadDump::doit() {
340340

341341
void VM_ThreadDump::snapshot_thread(JavaThread* java_thread, ThreadConcurrentLocks* tcl) {
342342
ThreadSnapshot* snapshot = _result->add_thread_snapshot(java_thread);
343-
snapshot->dump_stack_at_safepoint(_max_depth, _with_locked_monitors);
343+
snapshot->dump_stack_at_safepoint(_max_depth, _with_locked_monitors, false);
344344
snapshot->set_concurrent_locks(tcl);
345345
}
346346

‎src/hotspot/share/services/heapDumper.cpp

+1-1
Original file line numberDiff line numberDiff line change
@@ -1891,7 +1891,7 @@ void VM_HeapDumper::dump_stack_traces() {
18911891
// dump thread stack trace
18921892
ResourceMark rm;
18931893
ThreadStackTrace* stack_trace = new ThreadStackTrace(thread, false);
1894-
stack_trace->dump_stack_at_safepoint(-1);
1894+
stack_trace->dump_stack_at_safepoint(-1, true);
18951895
_stack_traces[_num_threads++] = stack_trace;
18961896

18971897
// write HPROF_FRAME records for this thread's stack trace

‎src/hotspot/share/services/threadService.cpp

+4-4
Original file line numberDiff line numberDiff line change
@@ -666,13 +666,13 @@ ThreadStackTrace::~ThreadStackTrace() {
666666
}
667667
}
668668

669-
void ThreadStackTrace::dump_stack_at_safepoint(int maxDepth) {
669+
void ThreadStackTrace::dump_stack_at_safepoint(int maxDepth, bool full) {
670670
assert(SafepointSynchronize::is_at_safepoint(), "all threads are stopped");
671671

672672
if (_thread->has_last_Java_frame()) {
673673
RegisterMap reg_map(_thread, true, false);
674674

675-
vframe* start_vf = _thread->last_continuation(java_lang_VirtualThread::vthread_scope()) != NULL
675+
vframe* start_vf = !full && _thread->last_continuation(java_lang_VirtualThread::vthread_scope()) != NULL
676676
? _thread->vthread_carrier_last_java_vframe(&reg_map)
677677
: _thread->last_java_vframe(&reg_map);
678678
int count = 0;
@@ -943,9 +943,9 @@ ThreadSnapshot::~ThreadSnapshot() {
943943
delete _concurrent_locks;
944944
}
945945

946-
void ThreadSnapshot::dump_stack_at_safepoint(int max_depth, bool with_locked_monitors) {
946+
void ThreadSnapshot::dump_stack_at_safepoint(int max_depth, bool with_locked_monitors, bool full) {
947947
_stack_trace = new ThreadStackTrace(_thread, with_locked_monitors);
948-
_stack_trace->dump_stack_at_safepoint(max_depth);
948+
_stack_trace->dump_stack_at_safepoint(max_depth, full);
949949
}
950950

951951

‎src/hotspot/share/services/threadService.hpp

+2-2
Original file line numberDiff line numberDiff line change
@@ -247,7 +247,7 @@ class ThreadSnapshot : public CHeapObj<mtInternal> {
247247
ThreadStackTrace* get_stack_trace() { return _stack_trace; }
248248
ThreadConcurrentLocks* get_concurrent_locks() { return _concurrent_locks; }
249249

250-
void dump_stack_at_safepoint(int max_depth, bool with_locked_monitors);
250+
void dump_stack_at_safepoint(int max_depth, bool with_locked_monitors, bool full);
251251
void set_concurrent_locks(ThreadConcurrentLocks* l) { _concurrent_locks = l; }
252252
void metadata_do(void f(Metadata*));
253253
};
@@ -270,7 +270,7 @@ class ThreadStackTrace : public CHeapObj<mtInternal> {
270270
int get_stack_depth() { return _depth; }
271271

272272
void add_stack_frame(javaVFrame* jvf);
273-
void dump_stack_at_safepoint(int max_depth);
273+
void dump_stack_at_safepoint(int max_depth, bool full);
274274
Handle allocate_fill_stack_trace_element_array(TRAPS);
275275
void metadata_do(void f(Metadata*));
276276
GrowableArray<OopHandle>* jni_locked_monitors() { return _jni_locked_monitors; }

‎src/java.base/share/classes/java/lang/a

-2,123
This file was deleted.

‎src/java.base/windows/classes/sun/nio/ch/PipeImpl.java

+23-6
Original file line numberDiff line numberDiff line change
@@ -31,6 +31,7 @@
3131
import java.io.IOException;
3232
import java.net.InetAddress;
3333
import java.net.InetSocketAddress;
34+
import java.net.StandardSocketOptions;
3435
import java.nio.*;
3536
import java.nio.channels.*;
3637
import java.nio.channels.spi.*;
@@ -55,16 +56,19 @@ class PipeImpl
5556
private static final Random RANDOM_NUMBER_GENERATOR = new SecureRandom();
5657

5758
// Source and sink channels
58-
private SourceChannel source;
59-
private SinkChannel sink;
59+
private final SourceChannel source;
60+
private final SinkChannel sink;
6061

61-
private class Initializer
62+
private static class Initializer
6263
implements PrivilegedExceptionAction<Void>
6364
{
6465

6566
private final SelectorProvider sp;
6667

67-
private IOException ioe = null;
68+
private IOException ioe;
69+
70+
SourceChannelImpl source;
71+
SinkChannelImpl sink;
6872

6973
private Initializer(SelectorProvider sp) {
7074
this.sp = sp;
@@ -166,12 +170,25 @@ public void run() {
166170
}
167171
}
168172

169-
PipeImpl(final SelectorProvider sp) throws IOException {
173+
PipeImpl(SelectorProvider sp, boolean delay) throws IOException {
174+
Initializer initializer = new Initializer(sp);
170175
try {
171-
AccessController.doPrivileged(new Initializer(sp));
176+
AccessController.doPrivileged(initializer);
172177
} catch (PrivilegedActionException x) {
173178
throw (IOException)x.getCause();
174179
}
180+
181+
if (!delay) {
182+
SocketChannel sc = initializer.sink.sc;
183+
sc.setOption(StandardSocketOptions.TCP_NODELAY, true);
184+
}
185+
186+
this.source = initializer.source;
187+
this.sink = initializer.sink;
188+
}
189+
190+
PipeImpl(SelectorProvider sp) throws IOException {
191+
this(sp, true);
175192
}
176193

177194
public SourceChannel source() {

‎src/java.base/windows/classes/sun/nio/ch/WEPollSelectorImpl.java

+6-11
Original file line numberDiff line numberDiff line change
@@ -38,17 +38,12 @@
3838
import java.util.HashMap;
3939
import java.util.Map;
4040
import java.util.function.Consumer;
41-
import jdk.internal.misc.Unsafe;
4241
import static sun.nio.ch.WEPoll.*;
4342

4443
/**
4544
* Windows wepoll based Selector implementation
4645
*/
4746
class WEPollSelectorImpl extends SelectorImpl {
48-
private static final Unsafe UNSAFE = Unsafe.getUnsafe();
49-
private static final long TEMP_BUF = UNSAFE.allocateMemory(1);
50-
private static final NativeDispatcher ND = new SocketDispatcher();
51-
5247
// maximum number of events to poll in one call to epoll_wait
5348
private static final int NUM_EPOLLEVENTS = 256;
5449

@@ -70,7 +65,7 @@ class WEPollSelectorImpl extends SelectorImpl {
7065
private boolean interruptTriggered;
7166
private final Pipe pipe;
7267
private final FileDescriptor fd0, fd1;
73-
private final int fd0Val;
68+
private final int fd0Val, fd1Val;
7469

7570
WEPollSelectorImpl(SelectorProvider sp) throws IOException {
7671
super(sp);
@@ -79,13 +74,13 @@ class WEPollSelectorImpl extends SelectorImpl {
7974
this.pollArrayAddress = WEPoll.allocatePollArray(NUM_EPOLLEVENTS);
8075

8176
// wakeup support
82-
this.pipe = new PipeImpl(null);
77+
this.pipe = new PipeImpl(null, /*no delay*/ false);
8378
SourceChannelImpl source = (SourceChannelImpl) pipe.source();
8479
SinkChannelImpl sink = (SinkChannelImpl) pipe.sink();
85-
(sink.sc).socket().setTcpNoDelay(true);
8680
this.fd0 = source.getFD();
87-
this.fd0Val = source.getFDVal();
8881
this.fd1 = sink.getFD();
82+
this.fd0Val = source.getFDVal();
83+
this.fd1Val = sink.getFDVal();
8984

9085
// register one end of the pipe for wakeups
9186
WEPoll.ctl(eph, EPOLL_CTL_ADD, fd0Val, WEPoll.EPOLLIN);
@@ -247,7 +242,7 @@ public Selector wakeup() {
247242
synchronized (interruptLock) {
248243
if (!interruptTriggered) {
249244
try {
250-
ND.write(fd1, TEMP_BUF, 1);
245+
IOUtil.write1(fd1Val, (byte) 0);
251246
} catch (IOException ioe) {
252247
throw new InternalError(ioe);
253248
}
@@ -259,7 +254,7 @@ public Selector wakeup() {
259254

260255
private void clearInterrupt() throws IOException {
261256
synchronized (interruptLock) {
262-
ND.read(fd0, TEMP_BUF, 1);
257+
IOUtil.drain(fd0Val);
263258
interruptTriggered = false;
264259
}
265260
}

‎src/java.base/windows/classes/sun/nio/ch/WSAPollSelectorImpl.java

+7-10
Original file line numberDiff line numberDiff line change
@@ -37,7 +37,6 @@
3737
import java.util.Deque;
3838
import java.util.List;
3939
import java.util.function.Consumer;
40-
import jdk.internal.misc.Unsafe;
4140

4241
/**
4342
* Selector implementation based on WASPoll.
@@ -48,10 +47,6 @@
4847
*/
4948

5049
class WSAPollSelectorImpl extends SelectorImpl {
51-
private static final Unsafe UNSAFE = Unsafe.getUnsafe();
52-
private static final long TEMP_BUF = UNSAFE.allocateMemory(1);
53-
private static final NativeDispatcher ND = new SocketDispatcher();
54-
5550
// initial capacity of poll array
5651
private static final int INITIAL_CAPACITY = 16;
5752

@@ -72,6 +67,7 @@ class WSAPollSelectorImpl extends SelectorImpl {
7267
private boolean interruptTriggered;
7368
private final Pipe pipe;
7469
private final FileDescriptor fd0, fd1;
70+
private final int fd0Val, fd1Val;
7571

7672
WSAPollSelectorImpl(SelectorProvider sp) throws IOException {
7773
super(sp);
@@ -80,16 +76,17 @@ class WSAPollSelectorImpl extends SelectorImpl {
8076
this.pollArrayCapacity = INITIAL_CAPACITY;
8177

8278
// wakeup support
83-
this.pipe = new PipeImpl(null);
79+
this.pipe = new PipeImpl(null, /*no delay*/ false);
8480
SourceChannelImpl source = (SourceChannelImpl) pipe.source();
8581
SinkChannelImpl sink = (SinkChannelImpl) pipe.sink();
86-
(sink.sc).socket().setTcpNoDelay(true);
8782
this.fd0 = source.getFD();
8883
this.fd1 = sink.getFD();
84+
this.fd0Val = source.getFDVal();
85+
this.fd1Val = sink.getFDVal();
8986

9087
// element 0 in poll array is for wakeup.
9188
synchronized (this) {
92-
putDescriptor(0, source.getFDVal());
89+
putDescriptor(0, fd0Val);
9390
putEvents(0, Net.POLLIN);
9491
putRevents(0, (short) 0);
9592
pollArraySize = 1;
@@ -240,7 +237,7 @@ public Selector wakeup() {
240237
synchronized (interruptLock) {
241238
if (!interruptTriggered) {
242239
try {
243-
ND.write(fd1, TEMP_BUF, 1);
240+
IOUtil.write1(fd1Val, (byte) 0);
244241
} catch (IOException ioe) {
245242
throw new InternalError(ioe);
246243
}
@@ -252,7 +249,7 @@ public Selector wakeup() {
252249

253250
private void clearInterrupt() throws IOException {
254251
synchronized (interruptLock) {
255-
ND.read(fd0, TEMP_BUF, 1);
252+
IOUtil.drain(fd0Val);
256253
interruptTriggered = false;
257254
}
258255
}

‎src/java.base/windows/classes/sun/nio/ch/WindowsSelectorImpl.java

+3-6
Original file line numberDiff line numberDiff line change
@@ -139,13 +139,10 @@ private static final class MapEntry {
139139
WindowsSelectorImpl(SelectorProvider sp) throws IOException {
140140
super(sp);
141141
pollWrapper = new PollArrayWrapper(INIT_CAP);
142-
wakeupPipe = Pipe.open();
143-
wakeupSourceFd = ((SelChImpl)wakeupPipe.source()).getFDVal();
144142

145-
// Disable the Nagle algorithm so that the wakeup is more immediate
146-
SinkChannelImpl sink = (SinkChannelImpl)wakeupPipe.sink();
147-
(sink.sc).socket().setTcpNoDelay(true);
148-
wakeupSinkFd = ((SelChImpl)sink).getFDVal();
143+
wakeupPipe = new PipeImpl(null, /*no delay*/ false);
144+
wakeupSourceFd = ((SelChImpl) wakeupPipe.source()).getFDVal();
145+
wakeupSinkFd = ((SelChImpl) wakeupPipe.sink()).getFDVal();
149146

150147
pollWrapper.addWakeupSocket(wakeupSourceFd, 0);
151148
}

‎src/java.base/windows/native/libnio/ch/IOUtil.c

+16-34
Original file line numberDiff line numberDiff line change
@@ -148,45 +148,27 @@ Java_sun_nio_ch_IOUtil_configureBlocking(JNIEnv *env, jclass clazz,
148148
}
149149
}
150150

151-
/* Note: Drain uses the int fd value. It is currently not called
152-
on windows.
153-
*/
154151
JNIEXPORT jboolean JNICALL
155152
Java_sun_nio_ch_IOUtil_drain(JNIEnv *env, jclass cl, jint fd)
156153
{
157-
DWORD read = 0;
158-
int totalRead = 0;
159-
BOOL result = 0;
160-
HANDLE h = (HANDLE)_get_osfhandle(fd);
161-
char buf[128];
162-
163-
if (h == INVALID_HANDLE_VALUE) {
164-
JNU_ThrowIOExceptionWithLastError(env, "Read failed");
165-
return JNI_FALSE;
166-
}
167-
154+
char buf[16];
155+
jboolean readBytes = JNI_FALSE;
168156
for (;;) {
169-
result = ReadFile(h, /* File handle to read */
170-
(LPVOID)&buf, /* address to put data */
171-
128, /* number of bytes to read */
172-
&read, /* number of bytes read */
173-
NULL); /* no overlapped struct */
174-
175-
if (result == 0) {
176-
int error = GetLastError();
177-
if (error == ERROR_NO_DATA) {
178-
return (totalRead > 0) ? JNI_TRUE : JNI_FALSE;
179-
}
180-
JNU_ThrowIOExceptionWithLastError(env, "Drain");
181-
return JNI_FALSE;
182-
}
183-
if (read > 0) {
184-
totalRead += read;
185-
} else {
186-
break;
187-
}
157+
int n = recv(fd, buf, sizeof(buf), 0);
158+
if (n <= 0)
159+
return readBytes;
160+
if (n < (int)sizeof(buf))
161+
return JNI_TRUE;
162+
readBytes = JNI_TRUE;
188163
}
189-
return (totalRead > 0) ? JNI_TRUE : JNI_FALSE;
164+
}
165+
166+
JNIEXPORT jint JNICALL
167+
Java_sun_nio_ch_IOUtil_write1(JNIEnv *env, jclass cl, jint fd, jbyte b)
168+
{
169+
SOCKET s = (SOCKET) fd;
170+
int n = send(s, &b, 1, 0);
171+
return (n == 1) ? 1 : 0;
190172
}
191173

192174
/* Note: This function returns the int fd value from file descriptor.

‎test/jdk/java/lang/Continuation/Fuzz.java

+110-74
Original file line numberDiff line numberDiff line change
@@ -21,8 +21,6 @@
2121
* questions.
2222
*/
2323

24-
25-
2624
/*
2725
* @test
2826
* @summary Fuzz tests for java.lang.Continuation
@@ -45,31 +43,19 @@
4543
// * @run main/othervm/timeout=300 -XX:+UseContinuationLazyCopy -XX:-UseContinuationChunks -XX:+UnlockDiagnosticVMOptions -XX:+WhiteBoxAPI -Xbootclasspath/a:. Fuzz
4644
// * @run main/othervm/timeout=300 -XX:+UseContinuationLazyCopy -XX:+UseContinuationChunks -XX:+UnlockDiagnosticVMOptions -XX:+WhiteBoxAPI -Xbootclasspath/a:. Fuzz
4745

46+
// * @run main/othervm/timeout=3000 -XX:StartFlightRecording=filename=test.jfr,settings=profile -XX:+UseContinuationLazyCopy -XX:-UseContinuationChunks -XX:+UnlockDiagnosticVMOptions -XX:+WhiteBoxAPI -Xbootclasspath/a:. Fuzz
47+
4848
// Anything excluded or not compileonly is not compiled; see CompilerOracle::should_exclude
4949

5050
// @run driver jdk.test.lib.FileInstaller compilerDirectives.json compilerDirectives.json
5151
// -XX:CompilerDirectivesFile=compilerDirectives.json
5252

53-
import java.lang.invoke.MethodHandle;
54-
import java.lang.invoke.MethodHandles;
55-
import java.lang.invoke.MethodType;
56-
import java.lang.reflect.InvocationTargetException;
57-
import java.lang.reflect.Method;
58-
import java.lang.reflect.Modifier;
53+
import java.lang.invoke.*;
54+
import java.lang.reflect.*;
5955
import java.lang.StackWalker.StackFrame;
6056
import java.io.IOException;
61-
import java.nio.file.Files;
62-
import java.nio.file.Path;
63-
import java.util.ArrayList;
64-
import java.util.Arrays;
65-
import java.util.EnumMap;
66-
import java.util.EnumSet;
67-
import java.util.List;
68-
import java.util.Map;
69-
import java.util.Objects;
70-
import java.util.Random;
71-
import java.util.Set;
72-
import java.util.HashSet;
57+
import java.nio.file.*;
58+
import java.util.*;
7359
import java.util.stream.Collectors;
7460
import java.util.stream.Stream;
7561
import static java.lang.Math.max;
@@ -80,54 +66,34 @@
8066
import sun.hotspot.WhiteBox;
8167

8268
public class Fuzz implements Runnable {
83-
static final boolean VERIFY_STACK = true; // could add significant time
69+
static final boolean VERIFY_STACK = false; // could add significant time
70+
static final boolean FILE = true;
8471
static final boolean RANDOM = false;
8572
static final boolean VERBOSE = false;
8673

87-
public static void main(String[] args) {
88-
for (int compileLevel : new int[]{4})
89-
for (boolean compileRun : new boolean[]{true})
90-
test(compileLevel, compileRun);
91-
}
92-
93-
private static boolean COMPILE_RUN;
94-
private static int COMPILE_LEVEL;
95-
96-
static void test(int compileLevel, boolean compileRun) {
97-
resetCompilation();
98-
COMPILE_LEVEL = compileLevel;
99-
COMPILE_RUN = compileRun;
74+
static final String FILENAME = "fuzz.dat";
75+
static final Path TEST_DIR = Path.of(System.getProperty("test.src", "."));
10076

101-
testFile();
102-
if (RANDOM)
103-
testRandom();
104-
}
105-
106-
static void testFile() {
107-
System.out.println("-- FILE --");
108-
try {
109-
testStream(file(Path.of(System.getProperty("test.src", ".")).resolve("fuzz.dat")));
110-
} catch (IOException e) { throw new RuntimeException(e); }
111-
}
112-
113-
static void testRandom() {
114-
long seed = 1L; // System.currentTimeMillis();
115-
System.out.println("-- RANDOM (seed: " + seed + ") --");
116-
testStream(random(new Random(seed)).limit(50));
77+
public static void main(String[] args) {
78+
warmup();
79+
for (int compileLevel : new int[]{4}) {
80+
for (boolean compileRun : new boolean[]{true}) {
81+
COMPILE_LEVEL = compileLevel;
82+
COMPILE_RUN = compileRun;
83+
resetCompilation();
84+
runTests();
85+
}
86+
}
11787
}
11888

119-
static Stream<Op[]> file(Path file) throws IOException {
120-
return Files.lines(file).map(String::trim).filter(s -> !s.isBlank() && !s.startsWith("#")).map(Fuzz::parse);
89+
static void runTests() {
90+
if (FILE) testFile(FILENAME);
91+
if (RANDOM) testRandom(System.currentTimeMillis(), 50);
12192
}
12293

123-
static Stream<Op[]> random(Random rnd) {
124-
var g = new Generator(rnd);
125-
return Stream.iterate(0, x->x+1).map(__ -> g.generate());
126-
}
94+
static void testStream(Stream<Op[]> traces) { traces.forEach(Fuzz::testTrace); }
12795

128-
static void testStream(Stream<Op[]> traces) {
129-
traces.forEach(Fuzz::testTrace);
130-
}
96+
static final int RETRIES = 4;
13197

13298
static void testTrace(Op[] trace) {
13399
System.out.println();
@@ -144,9 +110,9 @@ static void testTrace(Op[] trace) {
144110
int yields = fuzz.test();
145111
time(start, "Test (" + yields + " yields)");
146112

147-
if (!fuzz.checkCompilation("AFTER")) {
148-
if (retry++ < 2) {
149-
System.out.println("RETRYING");
113+
if (!fuzz.checkCompilation()) {
114+
if (retry++ < RETRIES) {
115+
System.out.println("RETRYING "+ retry);
150116
continue;
151117
}
152118
}
@@ -180,7 +146,7 @@ enum Op {
180146
static final Op[] ARRAY = new Op[0];
181147
}
182148

183-
///// Trace Gnereation
149+
///// Trace Generation
184150

185151
static class Generator {
186152
public Op[] generate() {
@@ -219,6 +185,33 @@ public Op[] generate() {
219185
private int plusOrMinus(int n) { return rnd.nextInt(2*n + 1) - n; }
220186
}
221187

188+
static Stream<Op[]> random(Random rnd) {
189+
var g = new Generator(rnd);
190+
return Stream.iterate(0, x->x+1).map(__ -> g.generate());
191+
}
192+
193+
static void testRandom(long seed, int number) {
194+
System.out.println("-- RANDOM (seed: " + seed + ") --");
195+
testStream(random(new Random(seed)).limit(number));
196+
}
197+
198+
//// File
199+
200+
static void testFile(String fileName) {
201+
System.out.println("-- FILE (" + FILENAME + ") --");
202+
try {
203+
testStream(file(TEST_DIR.resolve(fileName)));
204+
} catch (IOException e) { throw new RuntimeException(e); }
205+
}
206+
207+
static Stream<Op[]> file(Path file) throws IOException {
208+
return Files.lines(file).map(String::trim).filter(s -> !s.isBlank() && !s.startsWith("#")).map(Fuzz::parse);
209+
}
210+
211+
static Op[] parse(String line) {
212+
return Arrays.stream(line.split(", ")).map(s -> Enum.valueOf(Op.class, s))
213+
.collect(Collectors.toList()).toArray(Op.ARRAY);
214+
}
222215

223216
////////////////////////////////////////
224217

@@ -254,12 +247,14 @@ int test() {
254247
int count = 0;
255248
try {
256249
while (true) {
250+
var start = time();
257251
cont.run();
258252
if (cont.isDone()) break;
259253

260254
assert !shouldThrow();
261255
verifyStack(cont);
262256
count++;
257+
time(start, "Iteration");
263258
}
264259
verifyResult(result);
265260
} catch (FuzzException e) {
@@ -346,6 +341,7 @@ boolean shouldThrow() {
346341
switch (trace[i]) {
347342
case CALL_I_CTCH, CALL_C_CTCH -> { return false; }
348343
case THROW -> { return true; }
344+
default -> {}
349345
}
350346
}
351347
return false;
@@ -360,24 +356,28 @@ void captureStack() {
360356

361357
void verifyStack() {
362358
if (!VERIFY_STACK) return;
359+
var start = time();
363360
verifyStack(backtrace);
364361
verifyStack(backtrace, StackWalkerHelper.toStackTraceElement(fbacktrace));
365362
verifyStack(fbacktrace, lfbacktrace);
366363

367364
verifyStack(backtrace, Thread.currentThread().getStackTrace());
368365
verifyStack(fbacktrace, StackWalkerHelper.getStackFrames(SCOPE));
369366
verifyStack(lfbacktrace, StackWalkerHelper.getLiveStackFrames(SCOPE));
367+
time(start, "Verify stack");
370368
}
371369

372370
void verifyStack(Continuation cont) {
373371
if (!VERIFY_STACK) return;
372+
var start = time();
374373
verifyStack(backtrace);
375374
verifyStack(backtrace, StackWalkerHelper.toStackTraceElement(fbacktrace));
376375
verifyStack(fbacktrace, lfbacktrace);
377376

378377
verifyStack(backtrace, cont.getStackTrace());
379378
verifyStack(fbacktrace, StackWalkerHelper.getStackFrames(cont));
380379
verifyStack(lfbacktrace, StackWalkerHelper.getLiveStackFrames(cont));
380+
time(start, "Verify continuation stack");
381381
}
382382

383383
static boolean isStackCaptureMechanism(Object sf) {
@@ -452,7 +452,25 @@ static String sfToString(Object f) {
452452
return f instanceof StackFrame ? StackWalkerHelper.frameToString((StackFrame)f) : Objects.toString(f);
453453
}
454454

455-
////// Static Helpers
455+
////// Compilation
456+
457+
private static boolean COMPILE_RUN;
458+
private static int COMPILE_LEVEL;
459+
460+
static final int WARMUP_ITERS = 15_000;
461+
static final Op[] WARMUP_TRACE = {Op.MH_C_INT, Op.MH_C_MANY, Op.REF_C_INT, Op.REF_C_MANY, Op.CALL_C_INT};
462+
463+
static void warmup() {
464+
final long start = time();
465+
warmup(WARMUP_TRACE, WARMUP_ITERS); // generate (for reflection) and compile method handles
466+
time(start, "Warmup");
467+
}
468+
469+
static void warmup(Op[] trace, int times) {
470+
for (int i=0; i<times; i++) {
471+
new Fuzz(trace).run();
472+
}
473+
}
456474

457475
static void resetCompilation() {
458476
Set<Method> compile = Op.COMPILED.stream().map(Fuzz::method).collect(Collectors.toCollection(HashSet::new));
@@ -502,13 +520,34 @@ static void compile() {
502520
time(start, "Compile");
503521
}
504522

505-
boolean checkCompilation(String message) {
523+
boolean checkContinuationCompilation() {
524+
for (Method m : Continuation.class.getDeclaredMethods()) {
525+
if (!WB.isMethodCompiled(m)) {
526+
if (!Modifier.isNative(m.getModifiers())
527+
&& (m.getName().startsWith("enter")
528+
|| m.getName().startsWith("yield"))) {
529+
return false;
530+
}
531+
}
532+
}
533+
return true;
534+
}
535+
536+
boolean checkCompilation() {
537+
boolean res = true;
538+
539+
if (!checkContinuationCompilation()) {
540+
res = false;
541+
System.out.println("CHANGED CONTINUATION COMPILATION");
542+
}
543+
506544
Op[] newTrace = Arrays.copyOf(trace, trace.length);
507-
boolean res = checkCompilation(newTrace);
508-
if (!res) {
509-
System.out.println("CHANGED COMPILATION " + message);
545+
if (!checkCompilation(newTrace)) {
546+
res = false;
547+
System.out.println("CHANGED COMPILATION");
510548
printTrace(newTrace);
511549
}
550+
512551
return res;
513552
}
514553

@@ -523,6 +562,8 @@ static boolean checkCompilation(Op[] trace) {
523562
return ok;
524563
}
525564

565+
//// Static Helpers
566+
526567
static void rethrow(Throwable t) {
527568
if (t instanceof Error) throw (Error)t;
528569
if (t instanceof RuntimeException) throw (RuntimeException)t;
@@ -537,12 +578,7 @@ static <T> T[] arrayType(T[] array) {
537578

538579
static String write(Op[] trace) {
539580
return Arrays.stream(trace).map(Object::toString).collect(Collectors.joining(", "));
540-
}
541-
542-
static Op[] parse(String line) {
543-
return Arrays.stream(line.split(", ")).map(s -> Enum.valueOf(Op.class, s))
544-
.collect(Collectors.toList()).toArray(Op.ARRAY);
545-
}
581+
}
546582

547583
static Method method(Op op) { return method.get(op); }
548584
static MethodHandle handle(Op op) { return handle.get(op); }

0 commit comments

Comments
 (0)
Please sign in to comment.