blob: 84f208c9efb75d8a576826ad8cdc9fc1f425de49 (
plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
|
(***********************************************************************)
(* v * The Coq Proof Assistant / The Coq Development Team *)
(* <O___,, * INRIA-Rocquencourt & LRI-CNRS-Orsay *)
(* \VV/ *************************************************************)
(* // * This file is distributed under the terms of the *)
(* * GNU Lesser General Public License Version 2.1 *)
(***********************************************************************)
type raw_frame =
| Known_location of bool (* is_raise *)
* string (* filename *)
* int (* line number *)
* int (* start char *)
* int (* end char *)
| Unknown_location of bool (*is_raise*)
type location = {
loc_filename : string;
loc_line : int;
loc_start : int;
loc_end : int;
}
type frame = { frame_location : location option; frame_raised : bool; }
external get_exception_backtrace: unit -> raw_frame array option
= "caml_get_exception_backtrace"
type t = frame list option
let empty = Some []
let none = None
let of_raw = function
| Unknown_location r ->
{ frame_location = None; frame_raised = r; }
| Known_location (r, file, line, st, en) ->
let loc = {
loc_filename = file;
loc_line = line;
loc_start = st;
loc_end = en;
} in
{ frame_location = Some loc; frame_raised = r; }
let push bt = match bt with
| None -> None
| Some stack ->
match get_exception_backtrace () with
| None -> bt
| Some frames ->
let len = Array.length frames in
let rec append accu i =
if i = len then accu
else append (of_raw frames.(i) :: accu) (succ i)
in
Some (append stack 0)
(** Utilities *)
let print_frame frame =
let raise = if frame.frame_raised then "raise" else "frame" in
match frame.frame_location with
| None -> Printf.sprintf "%s @ unknown" raise
| Some loc ->
Printf.sprintf "%s @ file \"%s\", line %d, characters %d-%d"
raise loc.loc_filename loc.loc_line loc.loc_start loc.loc_end
(** Exception manipulation *)
let handlers = ref []
let register_backtrace_handler h =
handlers := h :: !handlers
let rec push_exn_aux e = function
| [] -> e
| f :: l ->
match f e with
| None -> push_exn_aux e l
| Some e -> e
let push_exn e =
push_exn_aux e !handlers
let reraise e =
let e = push_exn e in
raise e
|