(* From: https://github.com/msullivan/sml-util *)
(* Copyright (c) 2011-2015 Michael J. Sullivan *)
(* *)
(* Permission is hereby granted, free of charge, to any person obtaining a copy *)
(* of this software and associated documentation files (the "Software"), to deal *)
(* in the Software without restriction, including without limitation the rights *)
(* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell *)
(* copies of the Software, and to permit persons to whom the Software is *)
(* furnished to do so, subject to the following conditions: *)
(* *)
(* The above copyright notice and this permission notice shall be included in *)
(* all copies or substantial portions of the Software. *)
(* *)
(* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR *)
(* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, *)
(* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE *)
(* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER *)
(* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, *)
(* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN *)
(* THE SOFTWARE. *)
signature TIMEOUT =
sig
exception Timeout
(* Run a function with a timeout. *)
val runWithTimeout : Time.time -> ('a -> 'b) -> 'a -> 'b option
(* Run a function with a timeout, raising Timeout if it triggers *)
val runWithTimeoutExn : Time.time -> ('a -> 'b) -> 'a -> 'b
end
structure Timeout :> TIMEOUT =
struct
exception Timeout
fun finally f final =
f () before ignore (final ())
handle e => (final (); raise e)
fun runWithTimeout t f x =
let val timer = SMLofNJ.IntervalTimer.setIntTimer
fun cleanup () =
(timer NONE;
Signals.setHandler (Signals.sigALRM, Signals.IGNORE); ())
val ret = ref NONE
fun doit k =
let fun handler _ = k
val _ = Signals.setHandler (Signals.sigALRM,
Signals.HANDLER handler)
val () = timer (SOME t)
in ret := SOME (f x) end
val () = finally (fn () => SMLofNJ.Cont.callcc doit) cleanup
in !ret end
fun runWithTimeoutExn t f x =
case runWithTimeout t f x
of SOME x => x
| NONE => raise Timeout
end