P# Machines
can execute arbitrary C# code in their actions. This page discusses what happens if such an exception is not caught inside the action itself. As is usual in asynchronous programming, one should be careful with uncaught exception to make sure they don't go unnoticed.
In test mode (i.e., when running a test with PSharpTester
), all uncaught exceptions are an error and the test will fail. PSharpTester
will stop the execution at that point and report an error to the user. Simple.
In production mode (i.e., when running a P# program in production), the P# runtime intercepts any uncaught exception in a machine action. The exception is then delivered to the OnFailure
delegate of the runtime. At this point, it is the user's responsibility to take the appropriate action. For instance, the user can cause the program to crash and create a dump (for debugging later) as follows:
runtime.OnFailure += delegate (Exception exception)
{
Environment.FailFast(exception.ToString(), exception);
};
It is important that the user induces a crash to stop the program, after perhaps taking some cleanup actions. If the user doesn't do so, the runtime automatically enters a "failure" mode. Machines may continue running their current action, but no additional work will take place. The runtime stops all Enqueue
and Dequeue
operations in the program. Thus, messages sent will not be delivered, and already-received messages will not be dequeued. If the user wishes to suppress an exception and have the rest of the program continue normal operation, then it is best to catch the exception in the action itself using a usual try { } catch { }
block, or override the machine's OnException
method to handle exceptions in one place.
The runtime takes care to call the OnFailure
delegate with the stack intact. That is, if the user launches a debugger from inside the OnFailure
method, she will see the stack (with local variables) at the point the exception was thrown. However, this is not always the case. If the exception was thrown from an async
machine action, then the stack may have gotten unwound even before the runtime gets to know of the exception (because of the way async
continuations get compiled). We currently don't have a way around this limitation. But regardless, the exception is reported to the user.