src/Pure/Tools/debugger.ML
author wenzelm
Wed, 29 Jul 2015 13:34:04 +0200
changeset 60830 f56e189350b2
parent 60829 4b16b778ce0d
child 60834 781f1168d31e
permissions -rw-r--r--
separate channel for debugger output; clarified thread name;
Ignore whitespace changes - Everywhere: Within whitespace: At end of lines:
60749
f727b99faaf7 skeleton for interactive debugger;
wenzelm
parents:
diff changeset
     1
(*  Title:      Pure/Tools/debugger.ML
f727b99faaf7 skeleton for interactive debugger;
wenzelm
parents:
diff changeset
     2
    Author:     Makarius
f727b99faaf7 skeleton for interactive debugger;
wenzelm
parents:
diff changeset
     3
f727b99faaf7 skeleton for interactive debugger;
wenzelm
parents:
diff changeset
     4
Interactive debugger for Isabelle/ML.
f727b99faaf7 skeleton for interactive debugger;
wenzelm
parents:
diff changeset
     5
*)
f727b99faaf7 skeleton for interactive debugger;
wenzelm
parents:
diff changeset
     6
60830
f56e189350b2 separate channel for debugger output;
wenzelm
parents: 60829
diff changeset
     7
signature DEBUGGER =
f56e189350b2 separate channel for debugger output;
wenzelm
parents: 60829
diff changeset
     8
sig
f56e189350b2 separate channel for debugger output;
wenzelm
parents: 60829
diff changeset
     9
  val output: string -> unit
f56e189350b2 separate channel for debugger output;
wenzelm
parents: 60829
diff changeset
    10
end;
f56e189350b2 separate channel for debugger output;
wenzelm
parents: 60829
diff changeset
    11
f56e189350b2 separate channel for debugger output;
wenzelm
parents: 60829
diff changeset
    12
structure Debugger: DEBUGGER =
60765
e43e71a75838 support for ML debugger;
wenzelm
parents: 60749
diff changeset
    13
struct
e43e71a75838 support for ML debugger;
wenzelm
parents: 60749
diff changeset
    14
60830
f56e189350b2 separate channel for debugger output;
wenzelm
parents: 60829
diff changeset
    15
(* messages *)
f56e189350b2 separate channel for debugger output;
wenzelm
parents: 60829
diff changeset
    16
f56e189350b2 separate channel for debugger output;
wenzelm
parents: 60829
diff changeset
    17
fun output msg =
f56e189350b2 separate channel for debugger output;
wenzelm
parents: 60829
diff changeset
    18
  Output.protocol_message
f56e189350b2 separate channel for debugger output;
wenzelm
parents: 60829
diff changeset
    19
    (Markup.debugger_output (Simple_Thread.the_name ()) (serial ())) [msg];
f56e189350b2 separate channel for debugger output;
wenzelm
parents: 60829
diff changeset
    20
f56e189350b2 separate channel for debugger output;
wenzelm
parents: 60829
diff changeset
    21
60765
e43e71a75838 support for ML debugger;
wenzelm
parents: 60749
diff changeset
    22
(* global state *)
e43e71a75838 support for ML debugger;
wenzelm
parents: 60749
diff changeset
    23
e43e71a75838 support for ML debugger;
wenzelm
parents: 60749
diff changeset
    24
datatype state =
e43e71a75838 support for ML debugger;
wenzelm
parents: 60749
diff changeset
    25
  State of {
60829
4b16b778ce0d clarified thread name;
wenzelm
parents: 60765
diff changeset
    26
    threads: Thread.thread Symtab.table,  (*thread name ~> thread*)
4b16b778ce0d clarified thread name;
wenzelm
parents: 60765
diff changeset
    27
    input: string list Queue.T Symtab.table  (*thread name ~> input queue*)
60765
e43e71a75838 support for ML debugger;
wenzelm
parents: 60749
diff changeset
    28
  };
e43e71a75838 support for ML debugger;
wenzelm
parents: 60749
diff changeset
    29
e43e71a75838 support for ML debugger;
wenzelm
parents: 60749
diff changeset
    30
fun make_state (threads, input) = State {threads = threads, input = input};
e43e71a75838 support for ML debugger;
wenzelm
parents: 60749
diff changeset
    31
val init_state = make_state (Symtab.empty, Symtab.empty);
e43e71a75838 support for ML debugger;
wenzelm
parents: 60749
diff changeset
    32
fun map_state f (State {threads, input}) = make_state (f (threads, input));
e43e71a75838 support for ML debugger;
wenzelm
parents: 60749
diff changeset
    33
e43e71a75838 support for ML debugger;
wenzelm
parents: 60749
diff changeset
    34
val global_state = Synchronized.var "Debugger.state" init_state;
e43e71a75838 support for ML debugger;
wenzelm
parents: 60749
diff changeset
    35
60829
4b16b778ce0d clarified thread name;
wenzelm
parents: 60765
diff changeset
    36
fun cancel name =
60765
e43e71a75838 support for ML debugger;
wenzelm
parents: 60749
diff changeset
    37
  Synchronized.change global_state (tap (fn State {threads, ...} =>
60829
4b16b778ce0d clarified thread name;
wenzelm
parents: 60765
diff changeset
    38
    (case Symtab.lookup threads name of
60765
e43e71a75838 support for ML debugger;
wenzelm
parents: 60749
diff changeset
    39
      NONE => ()
e43e71a75838 support for ML debugger;
wenzelm
parents: 60749
diff changeset
    40
    | SOME thread => Simple_Thread.interrupt_unsynchronized thread)));
e43e71a75838 support for ML debugger;
wenzelm
parents: 60749
diff changeset
    41
60829
4b16b778ce0d clarified thread name;
wenzelm
parents: 60765
diff changeset
    42
fun input name msg =
60765
e43e71a75838 support for ML debugger;
wenzelm
parents: 60749
diff changeset
    43
  Synchronized.change global_state (map_state (fn (threads, input) =>
60829
4b16b778ce0d clarified thread name;
wenzelm
parents: 60765
diff changeset
    44
    let val input' = Symtab.map_default (name, Queue.empty) (Queue.enqueue msg) input;
60765
e43e71a75838 support for ML debugger;
wenzelm
parents: 60749
diff changeset
    45
    in (threads, input') end));
e43e71a75838 support for ML debugger;
wenzelm
parents: 60749
diff changeset
    46
60829
4b16b778ce0d clarified thread name;
wenzelm
parents: 60765
diff changeset
    47
fun get_input name =
60765
e43e71a75838 support for ML debugger;
wenzelm
parents: 60749
diff changeset
    48
  Synchronized.guarded_access global_state (fn State {threads, input} =>
60829
4b16b778ce0d clarified thread name;
wenzelm
parents: 60765
diff changeset
    49
    (case Symtab.lookup input name of
60765
e43e71a75838 support for ML debugger;
wenzelm
parents: 60749
diff changeset
    50
      NONE => NONE
e43e71a75838 support for ML debugger;
wenzelm
parents: 60749
diff changeset
    51
    | SOME queue =>
e43e71a75838 support for ML debugger;
wenzelm
parents: 60749
diff changeset
    52
        let
e43e71a75838 support for ML debugger;
wenzelm
parents: 60749
diff changeset
    53
          val (msg, queue') = Queue.dequeue queue;
e43e71a75838 support for ML debugger;
wenzelm
parents: 60749
diff changeset
    54
          val input' =
60829
4b16b778ce0d clarified thread name;
wenzelm
parents: 60765
diff changeset
    55
            if Queue.is_empty queue' then Symtab.delete_safe name input
4b16b778ce0d clarified thread name;
wenzelm
parents: 60765
diff changeset
    56
            else Symtab.update (name, queue') input;
60765
e43e71a75838 support for ML debugger;
wenzelm
parents: 60749
diff changeset
    57
        in SOME (msg, make_state (threads, input')) end));
e43e71a75838 support for ML debugger;
wenzelm
parents: 60749
diff changeset
    58
e43e71a75838 support for ML debugger;
wenzelm
parents: 60749
diff changeset
    59
e43e71a75838 support for ML debugger;
wenzelm
parents: 60749
diff changeset
    60
(* thread data *)
e43e71a75838 support for ML debugger;
wenzelm
parents: 60749
diff changeset
    61
e43e71a75838 support for ML debugger;
wenzelm
parents: 60749
diff changeset
    62
local val tag = Universal.tag () : ML_Debugger.state list Universal.tag in
e43e71a75838 support for ML debugger;
wenzelm
parents: 60749
diff changeset
    63
e43e71a75838 support for ML debugger;
wenzelm
parents: 60749
diff changeset
    64
fun get_data () = the_default [] (Thread.getLocal tag);
e43e71a75838 support for ML debugger;
wenzelm
parents: 60749
diff changeset
    65
fun setmp_data data f x = setmp_thread_data tag (get_data ()) data f x;
e43e71a75838 support for ML debugger;
wenzelm
parents: 60749
diff changeset
    66
60749
f727b99faaf7 skeleton for interactive debugger;
wenzelm
parents:
diff changeset
    67
end;
f727b99faaf7 skeleton for interactive debugger;
wenzelm
parents:
diff changeset
    68
60765
e43e71a75838 support for ML debugger;
wenzelm
parents: 60749
diff changeset
    69
val debugging = not o null o get_data;
e43e71a75838 support for ML debugger;
wenzelm
parents: 60749
diff changeset
    70
fun with_debugging e = setmp_data (ML_Debugger.state (Thread.self ())) e ();
e43e71a75838 support for ML debugger;
wenzelm
parents: 60749
diff changeset
    71
e43e71a75838 support for ML debugger;
wenzelm
parents: 60749
diff changeset
    72
e43e71a75838 support for ML debugger;
wenzelm
parents: 60749
diff changeset
    73
(* protocol messages *)
60749
f727b99faaf7 skeleton for interactive debugger;
wenzelm
parents:
diff changeset
    74
f727b99faaf7 skeleton for interactive debugger;
wenzelm
parents:
diff changeset
    75
val _ = Session.protocol_handler "isabelle.Debugger$Handler";
f727b99faaf7 skeleton for interactive debugger;
wenzelm
parents:
diff changeset
    76
60765
e43e71a75838 support for ML debugger;
wenzelm
parents: 60749
diff changeset
    77
e43e71a75838 support for ML debugger;
wenzelm
parents: 60749
diff changeset
    78
(* main entry point *)
e43e71a75838 support for ML debugger;
wenzelm
parents: 60749
diff changeset
    79
60829
4b16b778ce0d clarified thread name;
wenzelm
parents: 60765
diff changeset
    80
fun debug_loop name =
4b16b778ce0d clarified thread name;
wenzelm
parents: 60765
diff changeset
    81
  (case get_input name of
60765
e43e71a75838 support for ML debugger;
wenzelm
parents: 60749
diff changeset
    82
    ["continue"] => ()
e43e71a75838 support for ML debugger;
wenzelm
parents: 60749
diff changeset
    83
  | bad =>
e43e71a75838 support for ML debugger;
wenzelm
parents: 60749
diff changeset
    84
     (Output.system_message
e43e71a75838 support for ML debugger;
wenzelm
parents: 60749
diff changeset
    85
        ("Debugger: bad input " ^ ML_Syntax.print_list ML_Syntax.print_string bad);
60829
4b16b778ce0d clarified thread name;
wenzelm
parents: 60765
diff changeset
    86
      debug_loop name));
60765
e43e71a75838 support for ML debugger;
wenzelm
parents: 60749
diff changeset
    87
e43e71a75838 support for ML debugger;
wenzelm
parents: 60749
diff changeset
    88
fun debugger cond =
e43e71a75838 support for ML debugger;
wenzelm
parents: 60749
diff changeset
    89
  if debugging () orelse not (cond ()) orelse
e43e71a75838 support for ML debugger;
wenzelm
parents: 60749
diff changeset
    90
    not (Options.default_bool @{system_option ML_debugger_active})
e43e71a75838 support for ML debugger;
wenzelm
parents: 60749
diff changeset
    91
  then ()
e43e71a75838 support for ML debugger;
wenzelm
parents: 60749
diff changeset
    92
  else
e43e71a75838 support for ML debugger;
wenzelm
parents: 60749
diff changeset
    93
    with_debugging (fn () =>
60830
f56e189350b2 separate channel for debugger output;
wenzelm
parents: 60829
diff changeset
    94
      (case Simple_Thread.get_name () of
60765
e43e71a75838 support for ML debugger;
wenzelm
parents: 60749
diff changeset
    95
        NONE => ()
60829
4b16b778ce0d clarified thread name;
wenzelm
parents: 60765
diff changeset
    96
      | SOME name => debug_loop name));
60765
e43e71a75838 support for ML debugger;
wenzelm
parents: 60749
diff changeset
    97
e43e71a75838 support for ML debugger;
wenzelm
parents: 60749
diff changeset
    98
fun init () =
e43e71a75838 support for ML debugger;
wenzelm
parents: 60749
diff changeset
    99
  ML_Debugger.on_breakpoint
e43e71a75838 support for ML debugger;
wenzelm
parents: 60749
diff changeset
   100
    (SOME (fn (_, b) =>
e43e71a75838 support for ML debugger;
wenzelm
parents: 60749
diff changeset
   101
      debugger (fn () => ! b orelse Options.default_bool @{system_option ML_debugger_stepping})));
e43e71a75838 support for ML debugger;
wenzelm
parents: 60749
diff changeset
   102
e43e71a75838 support for ML debugger;
wenzelm
parents: 60749
diff changeset
   103
e43e71a75838 support for ML debugger;
wenzelm
parents: 60749
diff changeset
   104
(* protocol commands *)
e43e71a75838 support for ML debugger;
wenzelm
parents: 60749
diff changeset
   105
e43e71a75838 support for ML debugger;
wenzelm
parents: 60749
diff changeset
   106
val _ =
e43e71a75838 support for ML debugger;
wenzelm
parents: 60749
diff changeset
   107
  Isabelle_Process.protocol_command "Debugger.init"
e43e71a75838 support for ML debugger;
wenzelm
parents: 60749
diff changeset
   108
    (fn [] => init ());
e43e71a75838 support for ML debugger;
wenzelm
parents: 60749
diff changeset
   109
e43e71a75838 support for ML debugger;
wenzelm
parents: 60749
diff changeset
   110
val _ =
e43e71a75838 support for ML debugger;
wenzelm
parents: 60749
diff changeset
   111
  Isabelle_Process.protocol_command "Debugger.cancel"
60829
4b16b778ce0d clarified thread name;
wenzelm
parents: 60765
diff changeset
   112
    (fn [name] => cancel name);
60765
e43e71a75838 support for ML debugger;
wenzelm
parents: 60749
diff changeset
   113
e43e71a75838 support for ML debugger;
wenzelm
parents: 60749
diff changeset
   114
val _ =
e43e71a75838 support for ML debugger;
wenzelm
parents: 60749
diff changeset
   115
  Isabelle_Process.protocol_command "Debugger.input"
60829
4b16b778ce0d clarified thread name;
wenzelm
parents: 60765
diff changeset
   116
    (fn name :: msg => input name msg);
60765
e43e71a75838 support for ML debugger;
wenzelm
parents: 60749
diff changeset
   117
60749
f727b99faaf7 skeleton for interactive debugger;
wenzelm
parents:
diff changeset
   118
end;