2009-05-06 00:29:05 +00:00
|
|
|
/* Copyright (c) 2009, 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 avian;
|
|
|
|
|
|
|
|
import java.util.concurrent.Callable;
|
|
|
|
|
|
|
|
public abstract class Continuations {
|
|
|
|
public static native <T> T callWithCurrentContinuation
|
|
|
|
(CallbackReceiver<T> receiver) throws Exception;
|
|
|
|
|
2009-05-23 22:15:06 +00:00
|
|
|
public static <T> T dynamicWind(Runnable before,
|
|
|
|
Callable<T> thunk,
|
|
|
|
Runnable after)
|
|
|
|
throws Exception
|
|
|
|
{
|
2009-05-24 01:49:14 +00:00
|
|
|
UnwindResult result = dynamicWind2(before, thunk, after);
|
2009-05-23 22:15:06 +00:00
|
|
|
if (result.continuation != null) {
|
|
|
|
after.run();
|
|
|
|
if (result.exception != null) {
|
|
|
|
result.continuation.handleException(result.exception);
|
|
|
|
} else {
|
2009-05-24 01:49:14 +00:00
|
|
|
result.continuation.handleResult(result.result);
|
2009-05-23 22:15:06 +00:00
|
|
|
}
|
|
|
|
throw new AssertionError();
|
|
|
|
} else {
|
2009-05-24 01:49:14 +00:00
|
|
|
return (T) result.result;
|
2009-05-23 22:15:06 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private static native UnwindResult dynamicWind2(Runnable before,
|
2009-05-24 01:49:14 +00:00
|
|
|
Callable thunk,
|
2009-05-23 22:15:06 +00:00
|
|
|
Runnable after)
|
|
|
|
throws Exception;
|
|
|
|
|
|
|
|
private static UnwindResult wind(Runnable before,
|
2009-05-24 01:49:14 +00:00
|
|
|
Callable thunk,
|
2009-05-23 22:15:06 +00:00
|
|
|
Runnable after)
|
|
|
|
throws Exception
|
|
|
|
{
|
|
|
|
before.run();
|
|
|
|
|
|
|
|
try {
|
|
|
|
return new UnwindResult(null, thunk.call(), null);
|
|
|
|
} finally {
|
|
|
|
after.run();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private static void rewind(Runnable before,
|
|
|
|
Callback continuation,
|
|
|
|
Object result,
|
|
|
|
Throwable exception)
|
|
|
|
throws Exception
|
|
|
|
{
|
|
|
|
before.run();
|
|
|
|
|
|
|
|
if (exception != null) {
|
|
|
|
continuation.handleException(exception);
|
|
|
|
} else {
|
2009-05-24 01:49:14 +00:00
|
|
|
continuation.handleResult(result);
|
2009-05-23 22:15:06 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
throw new AssertionError();
|
|
|
|
}
|
2009-05-06 00:29:05 +00:00
|
|
|
|
|
|
|
private static class Continuation<T> implements Callback<T> {
|
|
|
|
public native void handleResult(T result);
|
|
|
|
public native void handleException(Throwable exception);
|
|
|
|
}
|
2009-05-23 22:15:06 +00:00
|
|
|
|
|
|
|
private static class UnwindResult {
|
|
|
|
public final Callback continuation;
|
|
|
|
public final Object result;
|
|
|
|
public final Throwable exception;
|
|
|
|
|
|
|
|
public UnwindResult(Callback continuation, Object result,
|
|
|
|
Throwable exception)
|
|
|
|
{
|
|
|
|
this.continuation = continuation;
|
|
|
|
this.result = result;
|
|
|
|
this.exception = exception;
|
|
|
|
}
|
|
|
|
}
|
2009-05-06 00:29:05 +00:00
|
|
|
}
|