src/Pure/Tools/debugger.scala
author wenzelm
Sat, 01 Apr 2017 19:17:15 +0200
changeset 65344 b99283eed13c
parent 65223 844c067bc3d4
child 71601 97ccf48c2f0c
permissions -rw-r--r--
clarified YXML vs. symbol encoding: operate on whole message;
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.scala
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
f727b99faaf7 skeleton for interactive debugger;
wenzelm
parents:
diff changeset
     7
package isabelle
f727b99faaf7 skeleton for interactive debugger;
wenzelm
parents:
diff changeset
     8
f727b99faaf7 skeleton for interactive debugger;
wenzelm
parents:
diff changeset
     9
61016
wenzelm
parents: 61014
diff changeset
    10
import scala.collection.immutable.SortedMap
wenzelm
parents: 61014
diff changeset
    11
wenzelm
parents: 61014
diff changeset
    12
60749
f727b99faaf7 skeleton for interactive debugger;
wenzelm
parents:
diff changeset
    13
object Debugger
f727b99faaf7 skeleton for interactive debugger;
wenzelm
parents:
diff changeset
    14
{
65213
51c0f094dc02 proper local debugger state, depending on session;
wenzelm
parents: 63805
diff changeset
    15
  /* thread context */
60834
781f1168d31e maintain debugger output messages;
wenzelm
parents: 60830
diff changeset
    16
65213
51c0f094dc02 proper local debugger state, depending on session;
wenzelm
parents: 63805
diff changeset
    17
  case object Update
51c0f094dc02 proper local debugger state, depending on session;
wenzelm
parents: 63805
diff changeset
    18
51c0f094dc02 proper local debugger state, depending on session;
wenzelm
parents: 63805
diff changeset
    19
  sealed case class Debug_State(pos: Position.T, function: String)
51c0f094dc02 proper local debugger state, depending on session;
wenzelm
parents: 63805
diff changeset
    20
  type Threads = SortedMap[String, List[Debug_State]]
60842
5510c8444bc4 protocol support for thread debugger state;
wenzelm
parents: 60835
diff changeset
    21
61010
cccfd7f6317d more explicit type Debugger.Context;
wenzelm
parents: 61008
diff changeset
    22
  sealed case class Context(thread_name: String, debug_states: List[Debug_State], index: Int = 0)
cccfd7f6317d more explicit type Debugger.Context;
wenzelm
parents: 61008
diff changeset
    23
  {
cccfd7f6317d more explicit type Debugger.Context;
wenzelm
parents: 61008
diff changeset
    24
    def size: Int = debug_states.length + 1
cccfd7f6317d more explicit type Debugger.Context;
wenzelm
parents: 61008
diff changeset
    25
    def reset: Context = copy(index = 0)
cccfd7f6317d more explicit type Debugger.Context;
wenzelm
parents: 61008
diff changeset
    26
    def select(i: Int) = copy(index = i + 1)
cccfd7f6317d more explicit type Debugger.Context;
wenzelm
parents: 61008
diff changeset
    27
cccfd7f6317d more explicit type Debugger.Context;
wenzelm
parents: 61008
diff changeset
    28
    def thread_state: Option[Debug_State] = debug_states.headOption
cccfd7f6317d more explicit type Debugger.Context;
wenzelm
parents: 61008
diff changeset
    29
cccfd7f6317d more explicit type Debugger.Context;
wenzelm
parents: 61008
diff changeset
    30
    def stack_state: Option[Debug_State] =
cccfd7f6317d more explicit type Debugger.Context;
wenzelm
parents: 61008
diff changeset
    31
      if (1 <= index && index <= debug_states.length)
cccfd7f6317d more explicit type Debugger.Context;
wenzelm
parents: 61008
diff changeset
    32
        Some(debug_states(index - 1))
cccfd7f6317d more explicit type Debugger.Context;
wenzelm
parents: 61008
diff changeset
    33
      else None
cccfd7f6317d more explicit type Debugger.Context;
wenzelm
parents: 61008
diff changeset
    34
61014
39f67bb4e609 maintain per-thread focus context;
wenzelm
parents: 61011
diff changeset
    35
    def debug_index: Option[Int] =
61010
cccfd7f6317d more explicit type Debugger.Context;
wenzelm
parents: 61008
diff changeset
    36
      if (stack_state.isDefined) Some(index - 1)
cccfd7f6317d more explicit type Debugger.Context;
wenzelm
parents: 61008
diff changeset
    37
      else if (debug_states.nonEmpty) Some(0)
cccfd7f6317d more explicit type Debugger.Context;
wenzelm
parents: 61008
diff changeset
    38
      else None
cccfd7f6317d more explicit type Debugger.Context;
wenzelm
parents: 61008
diff changeset
    39
cccfd7f6317d more explicit type Debugger.Context;
wenzelm
parents: 61008
diff changeset
    40
    def debug_state: Option[Debug_State] = stack_state orElse thread_state
61014
39f67bb4e609 maintain per-thread focus context;
wenzelm
parents: 61011
diff changeset
    41
    def debug_position: Option[Position.T] = debug_state.map(_.pos)
61010
cccfd7f6317d more explicit type Debugger.Context;
wenzelm
parents: 61008
diff changeset
    42
cccfd7f6317d more explicit type Debugger.Context;
wenzelm
parents: 61008
diff changeset
    43
    override def toString: String =
cccfd7f6317d more explicit type Debugger.Context;
wenzelm
parents: 61008
diff changeset
    44
      stack_state match {
cccfd7f6317d more explicit type Debugger.Context;
wenzelm
parents: 61008
diff changeset
    45
        case None => thread_name
cccfd7f6317d more explicit type Debugger.Context;
wenzelm
parents: 61008
diff changeset
    46
        case Some(d) => d.function
cccfd7f6317d more explicit type Debugger.Context;
wenzelm
parents: 61008
diff changeset
    47
      }
cccfd7f6317d more explicit type Debugger.Context;
wenzelm
parents: 61008
diff changeset
    48
  }
cccfd7f6317d more explicit type Debugger.Context;
wenzelm
parents: 61008
diff changeset
    49
cccfd7f6317d more explicit type Debugger.Context;
wenzelm
parents: 61008
diff changeset
    50
65213
51c0f094dc02 proper local debugger state, depending on session;
wenzelm
parents: 63805
diff changeset
    51
  /* debugger state */
61016
wenzelm
parents: 61014
diff changeset
    52
60835
6512bb0b1ff4 clarified management of (single) session;
wenzelm
parents: 60834
diff changeset
    53
  sealed case class State(
60932
13ee73f57c85 allow to break running threads at next possible breakpoint (simplified version of former option, see f3039309702e);
wenzelm
parents: 60912
diff changeset
    54
    active: Int = 0,  // active views
13ee73f57c85 allow to break running threads at next possible breakpoint (simplified version of former option, see f3039309702e);
wenzelm
parents: 60912
diff changeset
    55
    break: Boolean = false,  // break at next possible breakpoint
13ee73f57c85 allow to break running threads at next possible breakpoint (simplified version of former option, see f3039309702e);
wenzelm
parents: 60912
diff changeset
    56
    active_breakpoints: Set[Long] = Set.empty,  // explicit breakpoint state
61016
wenzelm
parents: 61014
diff changeset
    57
    threads: Threads = SortedMap.empty,  // thread name ~> stack of debug states
61014
39f67bb4e609 maintain per-thread focus context;
wenzelm
parents: 61011
diff changeset
    58
    focus: Map[String, Context] = Map.empty,  // thread name ~> focus
60834
781f1168d31e maintain debugger output messages;
wenzelm
parents: 60830
diff changeset
    59
    output: Map[String, Command.Results] = Map.empty)  // thread name ~> output messages
781f1168d31e maintain debugger output messages;
wenzelm
parents: 60830
diff changeset
    60
  {
60932
13ee73f57c85 allow to break running threads at next possible breakpoint (simplified version of former option, see f3039309702e);
wenzelm
parents: 60912
diff changeset
    61
    def set_break(b: Boolean): State = copy(break = b)
13ee73f57c85 allow to break running threads at next possible breakpoint (simplified version of former option, see f3039309702e);
wenzelm
parents: 60912
diff changeset
    62
65223
844c067bc3d4 more robust startup, despite remaining race condition of debugger.is_active vs. session.is_ready;
wenzelm
parents: 65222
diff changeset
    63
    def is_active: Boolean = active > 0
60898
a3fcde62df10 misc tuning and clarification;
wenzelm
parents: 60897
diff changeset
    64
    def inc_active: State = copy(active = active + 1)
a3fcde62df10 misc tuning and clarification;
wenzelm
parents: 60897
diff changeset
    65
    def dec_active: State = copy(active = active - 1)
60876
52edced9cce5 rendering for debugger/breakpoint active state;
wenzelm
parents: 60875
diff changeset
    66
60880
fa958e24ff24 set breakpoint state on ML side, relying on stable situation within the PIDE editing queue;
wenzelm
parents: 60878
diff changeset
    67
    def toggle_breakpoint(breakpoint: Long): (Boolean, State) =
60876
52edced9cce5 rendering for debugger/breakpoint active state;
wenzelm
parents: 60875
diff changeset
    68
    {
52edced9cce5 rendering for debugger/breakpoint active state;
wenzelm
parents: 60875
diff changeset
    69
      val active_breakpoints1 =
60880
fa958e24ff24 set breakpoint state on ML side, relying on stable situation within the PIDE editing queue;
wenzelm
parents: 60878
diff changeset
    70
        if (active_breakpoints(breakpoint)) active_breakpoints - breakpoint
fa958e24ff24 set breakpoint state on ML side, relying on stable situation within the PIDE editing queue;
wenzelm
parents: 60878
diff changeset
    71
      else active_breakpoints + breakpoint
fa958e24ff24 set breakpoint state on ML side, relying on stable situation within the PIDE editing queue;
wenzelm
parents: 60878
diff changeset
    72
      (active_breakpoints1(breakpoint), copy(active_breakpoints = active_breakpoints1))
60876
52edced9cce5 rendering for debugger/breakpoint active state;
wenzelm
parents: 60875
diff changeset
    73
    }
52edced9cce5 rendering for debugger/breakpoint active state;
wenzelm
parents: 60875
diff changeset
    74
60842
5510c8444bc4 protocol support for thread debugger state;
wenzelm
parents: 60835
diff changeset
    75
    def get_thread(thread_name: String): List[Debug_State] =
5510c8444bc4 protocol support for thread debugger state;
wenzelm
parents: 60835
diff changeset
    76
      threads.getOrElse(thread_name, Nil)
5510c8444bc4 protocol support for thread debugger state;
wenzelm
parents: 60835
diff changeset
    77
5510c8444bc4 protocol support for thread debugger state;
wenzelm
parents: 60835
diff changeset
    78
    def update_thread(thread_name: String, debug_states: List[Debug_State]): State =
61019
7ce030f14aa9 reset focus after thread update (with new debug_states);
wenzelm
parents: 61018
diff changeset
    79
    {
7ce030f14aa9 reset focus after thread update (with new debug_states);
wenzelm
parents: 61018
diff changeset
    80
      val threads1 =
7ce030f14aa9 reset focus after thread update (with new debug_states);
wenzelm
parents: 61018
diff changeset
    81
        if (debug_states.nonEmpty) threads + (thread_name -> debug_states)
7ce030f14aa9 reset focus after thread update (with new debug_states);
wenzelm
parents: 61018
diff changeset
    82
        else threads - thread_name
7ce030f14aa9 reset focus after thread update (with new debug_states);
wenzelm
parents: 61018
diff changeset
    83
      val focus1 =
7ce030f14aa9 reset focus after thread update (with new debug_states);
wenzelm
parents: 61018
diff changeset
    84
        focus.get(thread_name) match {
7ce030f14aa9 reset focus after thread update (with new debug_states);
wenzelm
parents: 61018
diff changeset
    85
          case Some(c) if debug_states.nonEmpty =>
7ce030f14aa9 reset focus after thread update (with new debug_states);
wenzelm
parents: 61018
diff changeset
    86
            focus + (thread_name -> Context(thread_name, debug_states))
7ce030f14aa9 reset focus after thread update (with new debug_states);
wenzelm
parents: 61018
diff changeset
    87
          case _ => focus - thread_name
7ce030f14aa9 reset focus after thread update (with new debug_states);
wenzelm
parents: 61018
diff changeset
    88
        }
7ce030f14aa9 reset focus after thread update (with new debug_states);
wenzelm
parents: 61018
diff changeset
    89
      copy(threads = threads1, focus = focus1)
7ce030f14aa9 reset focus after thread update (with new debug_states);
wenzelm
parents: 61018
diff changeset
    90
    }
60842
5510c8444bc4 protocol support for thread debugger state;
wenzelm
parents: 60835
diff changeset
    91
61014
39f67bb4e609 maintain per-thread focus context;
wenzelm
parents: 61011
diff changeset
    92
    def set_focus(c: Context): State = copy(focus = focus + (c.thread_name -> c))
39f67bb4e609 maintain per-thread focus context;
wenzelm
parents: 61011
diff changeset
    93
60842
5510c8444bc4 protocol support for thread debugger state;
wenzelm
parents: 60835
diff changeset
    94
    def get_output(thread_name: String): Command.Results =
5510c8444bc4 protocol support for thread debugger state;
wenzelm
parents: 60835
diff changeset
    95
      output.getOrElse(thread_name, Command.Results.empty)
5510c8444bc4 protocol support for thread debugger state;
wenzelm
parents: 60835
diff changeset
    96
5510c8444bc4 protocol support for thread debugger state;
wenzelm
parents: 60835
diff changeset
    97
    def add_output(thread_name: String, entry: Command.Results.Entry): State =
5510c8444bc4 protocol support for thread debugger state;
wenzelm
parents: 60835
diff changeset
    98
      copy(output = output + (thread_name -> (get_output(thread_name) + entry)))
5510c8444bc4 protocol support for thread debugger state;
wenzelm
parents: 60835
diff changeset
    99
5510c8444bc4 protocol support for thread debugger state;
wenzelm
parents: 60835
diff changeset
   100
    def clear_output(thread_name: String): State =
5510c8444bc4 protocol support for thread debugger state;
wenzelm
parents: 60835
diff changeset
   101
      copy(output = output - thread_name)
60834
781f1168d31e maintain debugger output messages;
wenzelm
parents: 60830
diff changeset
   102
  }
781f1168d31e maintain debugger output messages;
wenzelm
parents: 60830
diff changeset
   103
60902
9f185acfdcb8 clarified events;
wenzelm
parents: 60901
diff changeset
   104
9f185acfdcb8 clarified events;
wenzelm
parents: 60901
diff changeset
   105
  /* protocol handler */
9f185acfdcb8 clarified events;
wenzelm
parents: 60901
diff changeset
   106
65222
fb8253564483 more robust debugger initialization, e.g. required for GUI components before actual session startup;
wenzelm
parents: 65219
diff changeset
   107
  class Handler(session: Session) extends Session.Protocol_Handler
60749
f727b99faaf7 skeleton for interactive debugger;
wenzelm
parents:
diff changeset
   108
  {
65222
fb8253564483 more robust debugger initialization, e.g. required for GUI components before actual session startup;
wenzelm
parents: 65219
diff changeset
   109
    val debugger: Debugger = new Debugger(session)
65213
51c0f094dc02 proper local debugger state, depending on session;
wenzelm
parents: 63805
diff changeset
   110
65219
ed4b47b8c7dc misc tuning and simplification;
wenzelm
parents: 65218
diff changeset
   111
    private def debugger_state(msg: Prover.Protocol_Output): Boolean =
60842
5510c8444bc4 protocol support for thread debugger state;
wenzelm
parents: 60835
diff changeset
   112
    {
5510c8444bc4 protocol support for thread debugger state;
wenzelm
parents: 60835
diff changeset
   113
      msg.properties match {
5510c8444bc4 protocol support for thread debugger state;
wenzelm
parents: 60835
diff changeset
   114
        case Markup.Debugger_State(thread_name) =>
65344
b99283eed13c clarified YXML vs. symbol encoding: operate on whole message;
wenzelm
parents: 65223
diff changeset
   115
          val msg_body = Symbol.decode_yxml_failsafe(UTF8.decode_permissive(msg.bytes))
60842
5510c8444bc4 protocol support for thread debugger state;
wenzelm
parents: 60835
diff changeset
   116
          val debug_states =
5510c8444bc4 protocol support for thread debugger state;
wenzelm
parents: 60835
diff changeset
   117
          {
5510c8444bc4 protocol support for thread debugger state;
wenzelm
parents: 60835
diff changeset
   118
            import XML.Decode._
65344
b99283eed13c clarified YXML vs. symbol encoding: operate on whole message;
wenzelm
parents: 65223
diff changeset
   119
            list(pair(properties, string))(msg_body).map({
60842
5510c8444bc4 protocol support for thread debugger state;
wenzelm
parents: 60835
diff changeset
   120
              case (pos, function) => Debug_State(pos, function)
5510c8444bc4 protocol support for thread debugger state;
wenzelm
parents: 60835
diff changeset
   121
            })
5510c8444bc4 protocol support for thread debugger state;
wenzelm
parents: 60835
diff changeset
   122
          }
65222
fb8253564483 more robust debugger initialization, e.g. required for GUI components before actual session startup;
wenzelm
parents: 65219
diff changeset
   123
          debugger.update_thread(thread_name, debug_states)
60842
5510c8444bc4 protocol support for thread debugger state;
wenzelm
parents: 60835
diff changeset
   124
          true
5510c8444bc4 protocol support for thread debugger state;
wenzelm
parents: 60835
diff changeset
   125
        case _ => false
5510c8444bc4 protocol support for thread debugger state;
wenzelm
parents: 60835
diff changeset
   126
      }
5510c8444bc4 protocol support for thread debugger state;
wenzelm
parents: 60835
diff changeset
   127
    }
5510c8444bc4 protocol support for thread debugger state;
wenzelm
parents: 60835
diff changeset
   128
65219
ed4b47b8c7dc misc tuning and simplification;
wenzelm
parents: 65218
diff changeset
   129
    private def debugger_output(msg: Prover.Protocol_Output): Boolean =
60830
f56e189350b2 separate channel for debugger output;
wenzelm
parents: 60829
diff changeset
   130
    {
f56e189350b2 separate channel for debugger output;
wenzelm
parents: 60829
diff changeset
   131
      msg.properties match {
60835
6512bb0b1ff4 clarified management of (single) session;
wenzelm
parents: 60834
diff changeset
   132
        case Markup.Debugger_Output(thread_name) =>
65344
b99283eed13c clarified YXML vs. symbol encoding: operate on whole message;
wenzelm
parents: 65223
diff changeset
   133
          Symbol.decode_yxml_failsafe(UTF8.decode_permissive(msg.bytes)) match {
60834
781f1168d31e maintain debugger output messages;
wenzelm
parents: 60830
diff changeset
   134
            case List(XML.Elem(Markup(name, props @ Markup.Serial(i)), body)) =>
65218
102b8e092860 more explicit Session.xml_cache;
wenzelm
parents: 65217
diff changeset
   135
              val message = XML.Elem(Markup(Markup.message(name), props), body)
65222
fb8253564483 more robust debugger initialization, e.g. required for GUI components before actual session startup;
wenzelm
parents: 65219
diff changeset
   136
              debugger.add_output(thread_name, i -> session.xml_cache.elem(message))
60834
781f1168d31e maintain debugger output messages;
wenzelm
parents: 60830
diff changeset
   137
              true
781f1168d31e maintain debugger output messages;
wenzelm
parents: 60830
diff changeset
   138
            case _ => false
781f1168d31e maintain debugger output messages;
wenzelm
parents: 60830
diff changeset
   139
          }
60830
f56e189350b2 separate channel for debugger output;
wenzelm
parents: 60829
diff changeset
   140
        case _ => false
f56e189350b2 separate channel for debugger output;
wenzelm
parents: 60829
diff changeset
   141
      }
f56e189350b2 separate channel for debugger output;
wenzelm
parents: 60829
diff changeset
   142
    }
f56e189350b2 separate channel for debugger output;
wenzelm
parents: 60829
diff changeset
   143
f56e189350b2 separate channel for debugger output;
wenzelm
parents: 60829
diff changeset
   144
    val functions =
65219
ed4b47b8c7dc misc tuning and simplification;
wenzelm
parents: 65218
diff changeset
   145
      List(
60842
5510c8444bc4 protocol support for thread debugger state;
wenzelm
parents: 60835
diff changeset
   146
        Markup.DEBUGGER_STATE -> debugger_state _,
5510c8444bc4 protocol support for thread debugger state;
wenzelm
parents: 60835
diff changeset
   147
        Markup.DEBUGGER_OUTPUT -> debugger_output _)
60749
f727b99faaf7 skeleton for interactive debugger;
wenzelm
parents:
diff changeset
   148
  }
65213
51c0f094dc02 proper local debugger state, depending on session;
wenzelm
parents: 63805
diff changeset
   149
}
51c0f094dc02 proper local debugger state, depending on session;
wenzelm
parents: 63805
diff changeset
   150
65222
fb8253564483 more robust debugger initialization, e.g. required for GUI components before actual session startup;
wenzelm
parents: 65219
diff changeset
   151
class Debugger private(session: Session)
65213
51c0f094dc02 proper local debugger state, depending on session;
wenzelm
parents: 63805
diff changeset
   152
{
51c0f094dc02 proper local debugger state, depending on session;
wenzelm
parents: 63805
diff changeset
   153
  /* debugger state */
51c0f094dc02 proper local debugger state, depending on session;
wenzelm
parents: 63805
diff changeset
   154
65223
844c067bc3d4 more robust startup, despite remaining race condition of debugger.is_active vs. session.is_ready;
wenzelm
parents: 65222
diff changeset
   155
  private val state = Synchronized(Debugger.State())
65213
51c0f094dc02 proper local debugger state, depending on session;
wenzelm
parents: 63805
diff changeset
   156
51c0f094dc02 proper local debugger state, depending on session;
wenzelm
parents: 63805
diff changeset
   157
  private val delay_update =
51c0f094dc02 proper local debugger state, depending on session;
wenzelm
parents: 63805
diff changeset
   158
    Standard_Thread.delay_first(session.output_delay) {
51c0f094dc02 proper local debugger state, depending on session;
wenzelm
parents: 63805
diff changeset
   159
      session.debugger_updates.post(Debugger.Update)
51c0f094dc02 proper local debugger state, depending on session;
wenzelm
parents: 63805
diff changeset
   160
    }
60765
e43e71a75838 support for ML debugger;
wenzelm
parents: 60749
diff changeset
   161
e43e71a75838 support for ML debugger;
wenzelm
parents: 60749
diff changeset
   162
e43e71a75838 support for ML debugger;
wenzelm
parents: 60749
diff changeset
   163
  /* protocol commands */
e43e71a75838 support for ML debugger;
wenzelm
parents: 60749
diff changeset
   164
65213
51c0f094dc02 proper local debugger state, depending on session;
wenzelm
parents: 63805
diff changeset
   165
  def update_thread(thread_name: String, debug_states: List[Debugger.Debug_State])
60910
79abcf48c377 clarified init/exit vs. session phase;
wenzelm
parents: 60903
diff changeset
   166
  {
65213
51c0f094dc02 proper local debugger state, depending on session;
wenzelm
parents: 63805
diff changeset
   167
    state.change(_.update_thread(thread_name, debug_states))
51c0f094dc02 proper local debugger state, depending on session;
wenzelm
parents: 63805
diff changeset
   168
    delay_update.invoke()
60910
79abcf48c377 clarified init/exit vs. session phase;
wenzelm
parents: 60903
diff changeset
   169
  }
79abcf48c377 clarified init/exit vs. session phase;
wenzelm
parents: 60903
diff changeset
   170
65213
51c0f094dc02 proper local debugger state, depending on session;
wenzelm
parents: 63805
diff changeset
   171
  def add_output(thread_name: String, entry: Command.Results.Entry)
51c0f094dc02 proper local debugger state, depending on session;
wenzelm
parents: 63805
diff changeset
   172
  {
51c0f094dc02 proper local debugger state, depending on session;
wenzelm
parents: 63805
diff changeset
   173
    state.change(_.add_output(thread_name, entry))
51c0f094dc02 proper local debugger state, depending on session;
wenzelm
parents: 63805
diff changeset
   174
    delay_update.invoke()
51c0f094dc02 proper local debugger state, depending on session;
wenzelm
parents: 63805
diff changeset
   175
  }
51c0f094dc02 proper local debugger state, depending on session;
wenzelm
parents: 63805
diff changeset
   176
65223
844c067bc3d4 more robust startup, despite remaining race condition of debugger.is_active vs. session.is_ready;
wenzelm
parents: 65222
diff changeset
   177
  def is_active(): Boolean = session.is_ready && state.value.is_active
844c067bc3d4 more robust startup, despite remaining race condition of debugger.is_active vs. session.is_ready;
wenzelm
parents: 65222
diff changeset
   178
844c067bc3d4 more robust startup, despite remaining race condition of debugger.is_active vs. session.is_ready;
wenzelm
parents: 65222
diff changeset
   179
  def ready()
844c067bc3d4 more robust startup, despite remaining race condition of debugger.is_active vs. session.is_ready;
wenzelm
parents: 65222
diff changeset
   180
  {
844c067bc3d4 more robust startup, despite remaining race condition of debugger.is_active vs. session.is_ready;
wenzelm
parents: 65222
diff changeset
   181
    if (is_active())
844c067bc3d4 more robust startup, despite remaining race condition of debugger.is_active vs. session.is_ready;
wenzelm
parents: 65222
diff changeset
   182
      session.protocol_command("Debugger.init")
844c067bc3d4 more robust startup, despite remaining race condition of debugger.is_active vs. session.is_ready;
wenzelm
parents: 65222
diff changeset
   183
  }
65213
51c0f094dc02 proper local debugger state, depending on session;
wenzelm
parents: 63805
diff changeset
   184
60910
79abcf48c377 clarified init/exit vs. session phase;
wenzelm
parents: 60903
diff changeset
   185
  def init(): Unit =
65213
51c0f094dc02 proper local debugger state, depending on session;
wenzelm
parents: 63805
diff changeset
   186
    state.change(st =>
60889
7f210887cc4e init/exit depending on active debugger panels;
wenzelm
parents: 60888
diff changeset
   187
    {
65213
51c0f094dc02 proper local debugger state, depending on session;
wenzelm
parents: 63805
diff changeset
   188
      val st1 = st.inc_active
65223
844c067bc3d4 more robust startup, despite remaining race condition of debugger.is_active vs. session.is_ready;
wenzelm
parents: 65222
diff changeset
   189
      if (session.is_ready && !st.is_active && st1.is_active)
65213
51c0f094dc02 proper local debugger state, depending on session;
wenzelm
parents: 63805
diff changeset
   190
        session.protocol_command("Debugger.init")
51c0f094dc02 proper local debugger state, depending on session;
wenzelm
parents: 63805
diff changeset
   191
      st1
60889
7f210887cc4e init/exit depending on active debugger panels;
wenzelm
parents: 60888
diff changeset
   192
    })
7f210887cc4e init/exit depending on active debugger panels;
wenzelm
parents: 60888
diff changeset
   193
60910
79abcf48c377 clarified init/exit vs. session phase;
wenzelm
parents: 60903
diff changeset
   194
  def exit(): Unit =
65213
51c0f094dc02 proper local debugger state, depending on session;
wenzelm
parents: 63805
diff changeset
   195
    state.change(st =>
60889
7f210887cc4e init/exit depending on active debugger panels;
wenzelm
parents: 60888
diff changeset
   196
    {
65213
51c0f094dc02 proper local debugger state, depending on session;
wenzelm
parents: 63805
diff changeset
   197
      val st1 = st.dec_active
65223
844c067bc3d4 more robust startup, despite remaining race condition of debugger.is_active vs. session.is_ready;
wenzelm
parents: 65222
diff changeset
   198
      if (session.is_ready && st.is_active && !st1.is_active)
65213
51c0f094dc02 proper local debugger state, depending on session;
wenzelm
parents: 63805
diff changeset
   199
        session.protocol_command("Debugger.exit")
51c0f094dc02 proper local debugger state, depending on session;
wenzelm
parents: 63805
diff changeset
   200
      st1
60889
7f210887cc4e init/exit depending on active debugger panels;
wenzelm
parents: 60888
diff changeset
   201
    })
60876
52edced9cce5 rendering for debugger/breakpoint active state;
wenzelm
parents: 60875
diff changeset
   202
65213
51c0f094dc02 proper local debugger state, depending on session;
wenzelm
parents: 63805
diff changeset
   203
  def is_break(): Boolean = state.value.break
60932
13ee73f57c85 allow to break running threads at next possible breakpoint (simplified version of former option, see f3039309702e);
wenzelm
parents: 60912
diff changeset
   204
  def set_break(b: Boolean)
13ee73f57c85 allow to break running threads at next possible breakpoint (simplified version of former option, see f3039309702e);
wenzelm
parents: 60912
diff changeset
   205
  {
65213
51c0f094dc02 proper local debugger state, depending on session;
wenzelm
parents: 63805
diff changeset
   206
    state.change(st => {
51c0f094dc02 proper local debugger state, depending on session;
wenzelm
parents: 63805
diff changeset
   207
      val st1 = st.set_break(b)
51c0f094dc02 proper local debugger state, depending on session;
wenzelm
parents: 63805
diff changeset
   208
      session.protocol_command("Debugger.break", b.toString)
51c0f094dc02 proper local debugger state, depending on session;
wenzelm
parents: 63805
diff changeset
   209
      st1
60932
13ee73f57c85 allow to break running threads at next possible breakpoint (simplified version of former option, see f3039309702e);
wenzelm
parents: 60912
diff changeset
   210
    })
13ee73f57c85 allow to break running threads at next possible breakpoint (simplified version of former option, see f3039309702e);
wenzelm
parents: 60912
diff changeset
   211
    delay_update.invoke()
13ee73f57c85 allow to break running threads at next possible breakpoint (simplified version of former option, see f3039309702e);
wenzelm
parents: 60912
diff changeset
   212
  }
13ee73f57c85 allow to break running threads at next possible breakpoint (simplified version of former option, see f3039309702e);
wenzelm
parents: 60912
diff changeset
   213
60882
45bfd18835f1 tuned signature;
wenzelm
parents: 60880
diff changeset
   214
  def active_breakpoint_state(breakpoint: Long): Option[Boolean] =
60876
52edced9cce5 rendering for debugger/breakpoint active state;
wenzelm
parents: 60875
diff changeset
   215
  {
65213
51c0f094dc02 proper local debugger state, depending on session;
wenzelm
parents: 63805
diff changeset
   216
    val st = state.value
51c0f094dc02 proper local debugger state, depending on session;
wenzelm
parents: 63805
diff changeset
   217
    if (st.is_active) Some(st.active_breakpoints(breakpoint)) else None
60876
52edced9cce5 rendering for debugger/breakpoint active state;
wenzelm
parents: 60875
diff changeset
   218
  }
52edced9cce5 rendering for debugger/breakpoint active state;
wenzelm
parents: 60875
diff changeset
   219
60882
45bfd18835f1 tuned signature;
wenzelm
parents: 60880
diff changeset
   220
  def breakpoint_state(breakpoint: Long): Boolean =
65213
51c0f094dc02 proper local debugger state, depending on session;
wenzelm
parents: 63805
diff changeset
   221
    state.value.active_breakpoints(breakpoint)
60882
45bfd18835f1 tuned signature;
wenzelm
parents: 60880
diff changeset
   222
60880
fa958e24ff24 set breakpoint state on ML side, relying on stable situation within the PIDE editing queue;
wenzelm
parents: 60878
diff changeset
   223
  def toggle_breakpoint(command: Command, breakpoint: Long)
60878
1f0d2bbcf38b added action to toggle breakpoints (on editor side);
wenzelm
parents: 60876
diff changeset
   224
  {
65213
51c0f094dc02 proper local debugger state, depending on session;
wenzelm
parents: 63805
diff changeset
   225
    state.change(st =>
51c0f094dc02 proper local debugger state, depending on session;
wenzelm
parents: 63805
diff changeset
   226
      {
51c0f094dc02 proper local debugger state, depending on session;
wenzelm
parents: 63805
diff changeset
   227
        val (breakpoint_state, st1) = st.toggle_breakpoint(breakpoint)
51c0f094dc02 proper local debugger state, depending on session;
wenzelm
parents: 63805
diff changeset
   228
        session.protocol_command(
51c0f094dc02 proper local debugger state, depending on session;
wenzelm
parents: 63805
diff changeset
   229
          "Debugger.breakpoint",
51c0f094dc02 proper local debugger state, depending on session;
wenzelm
parents: 63805
diff changeset
   230
          Symbol.encode(command.node_name.node),
51c0f094dc02 proper local debugger state, depending on session;
wenzelm
parents: 63805
diff changeset
   231
          Document_ID(command.id),
51c0f094dc02 proper local debugger state, depending on session;
wenzelm
parents: 63805
diff changeset
   232
          Value.Long(breakpoint),
51c0f094dc02 proper local debugger state, depending on session;
wenzelm
parents: 63805
diff changeset
   233
          Value.Boolean(breakpoint_state))
51c0f094dc02 proper local debugger state, depending on session;
wenzelm
parents: 63805
diff changeset
   234
        st1
51c0f094dc02 proper local debugger state, depending on session;
wenzelm
parents: 63805
diff changeset
   235
      })
60878
1f0d2bbcf38b added action to toggle breakpoints (on editor side);
wenzelm
parents: 60876
diff changeset
   236
  }
60876
52edced9cce5 rendering for debugger/breakpoint active state;
wenzelm
parents: 60875
diff changeset
   237
65213
51c0f094dc02 proper local debugger state, depending on session;
wenzelm
parents: 63805
diff changeset
   238
  def status(focus: Option[Debugger.Context]): (Debugger.Threads, List[XML.Tree]) =
61018
32cc7d219c38 atomic Debugger.status;
wenzelm
parents: 61016
diff changeset
   239
  {
65213
51c0f094dc02 proper local debugger state, depending on session;
wenzelm
parents: 63805
diff changeset
   240
    val st = state.value
61018
32cc7d219c38 atomic Debugger.status;
wenzelm
parents: 61016
diff changeset
   241
    val output =
32cc7d219c38 atomic Debugger.status;
wenzelm
parents: 61016
diff changeset
   242
      focus match {
32cc7d219c38 atomic Debugger.status;
wenzelm
parents: 61016
diff changeset
   243
        case None => Nil
32cc7d219c38 atomic Debugger.status;
wenzelm
parents: 61016
diff changeset
   244
        case Some(c) =>
32cc7d219c38 atomic Debugger.status;
wenzelm
parents: 61016
diff changeset
   245
          (for {
65213
51c0f094dc02 proper local debugger state, depending on session;
wenzelm
parents: 63805
diff changeset
   246
            (thread_name, results) <- st.output
61018
32cc7d219c38 atomic Debugger.status;
wenzelm
parents: 61016
diff changeset
   247
            if thread_name == c.thread_name
32cc7d219c38 atomic Debugger.status;
wenzelm
parents: 61016
diff changeset
   248
            (_, tree) <- results.iterator
32cc7d219c38 atomic Debugger.status;
wenzelm
parents: 61016
diff changeset
   249
          } yield tree).toList
32cc7d219c38 atomic Debugger.status;
wenzelm
parents: 61016
diff changeset
   250
      }
65213
51c0f094dc02 proper local debugger state, depending on session;
wenzelm
parents: 63805
diff changeset
   251
    (st.threads, output)
61018
32cc7d219c38 atomic Debugger.status;
wenzelm
parents: 61016
diff changeset
   252
  }
61011
018b0c996b54 more explicit debugger caret rendering;
wenzelm
parents: 61010
diff changeset
   253
65213
51c0f094dc02 proper local debugger state, depending on session;
wenzelm
parents: 63805
diff changeset
   254
  def focus(): List[Debugger.Context] = state.value.focus.toList.map(_._2)
51c0f094dc02 proper local debugger state, depending on session;
wenzelm
parents: 63805
diff changeset
   255
  def set_focus(c: Debugger.Context)
61008
b88b8227e1a3 proper GUI event;
wenzelm
parents: 61007
diff changeset
   256
  {
65213
51c0f094dc02 proper local debugger state, depending on session;
wenzelm
parents: 63805
diff changeset
   257
    state.change(_.set_focus(c))
61008
b88b8227e1a3 proper GUI event;
wenzelm
parents: 61007
diff changeset
   258
    delay_update.invoke()
b88b8227e1a3 proper GUI event;
wenzelm
parents: 61007
diff changeset
   259
  }
60875
ee23c1d21ac3 follow debugger focus;
wenzelm
parents: 60869
diff changeset
   260
60854
8f45dd297357 more controls;
wenzelm
parents: 60842
diff changeset
   261
  def input(thread_name: String, msg: String*): Unit =
65213
51c0f094dc02 proper local debugger state, depending on session;
wenzelm
parents: 63805
diff changeset
   262
    session.protocol_command("Debugger.input", (thread_name :: msg.toList):_*)
60854
8f45dd297357 more controls;
wenzelm
parents: 60842
diff changeset
   263
60899
wenzelm
parents: 60898
diff changeset
   264
  def continue(thread_name: String): Unit = input(thread_name, "continue")
60869
878e32cf3131 more single stepping;
wenzelm
parents: 60863
diff changeset
   265
  def step(thread_name: String): Unit = input(thread_name, "step")
878e32cf3131 more single stepping;
wenzelm
parents: 60863
diff changeset
   266
  def step_over(thread_name: String): Unit = input(thread_name, "step_over")
878e32cf3131 more single stepping;
wenzelm
parents: 60863
diff changeset
   267
  def step_out(thread_name: String): Unit = input(thread_name, "step_out")
60856
eb21ae05ec43 clarified thread state;
wenzelm
parents: 60854
diff changeset
   268
60901
ce8abd005c5d clarified GUI event handling;
wenzelm
parents: 60900
diff changeset
   269
  def clear_output(thread_name: String)
ce8abd005c5d clarified GUI event handling;
wenzelm
parents: 60900
diff changeset
   270
  {
65213
51c0f094dc02 proper local debugger state, depending on session;
wenzelm
parents: 63805
diff changeset
   271
    state.change(_.clear_output(thread_name))
60902
9f185acfdcb8 clarified events;
wenzelm
parents: 60901
diff changeset
   272
    delay_update.invoke()
60901
ce8abd005c5d clarified GUI event handling;
wenzelm
parents: 60900
diff changeset
   273
  }
ce8abd005c5d clarified GUI event handling;
wenzelm
parents: 60900
diff changeset
   274
65213
51c0f094dc02 proper local debugger state, depending on session;
wenzelm
parents: 63805
diff changeset
   275
  def eval(c: Debugger.Context, SML: Boolean, context: String, expression: String)
60856
eb21ae05ec43 clarified thread state;
wenzelm
parents: 60854
diff changeset
   276
  {
65213
51c0f094dc02 proper local debugger state, depending on session;
wenzelm
parents: 63805
diff changeset
   277
    state.change(st => {
61014
39f67bb4e609 maintain per-thread focus context;
wenzelm
parents: 61011
diff changeset
   278
      input(c.thread_name, "eval", c.debug_index.getOrElse(0).toString,
61010
cccfd7f6317d more explicit type Debugger.Context;
wenzelm
parents: 61008
diff changeset
   279
        SML.toString, Symbol.encode(context), Symbol.encode(expression))
65213
51c0f094dc02 proper local debugger state, depending on session;
wenzelm
parents: 63805
diff changeset
   280
      st.clear_output(c.thread_name)
60896
625f2c8307da clarified output;
wenzelm
parents: 60889
diff changeset
   281
    })
60902
9f185acfdcb8 clarified events;
wenzelm
parents: 60901
diff changeset
   282
    delay_update.invoke()
60856
eb21ae05ec43 clarified thread state;
wenzelm
parents: 60854
diff changeset
   283
  }
60897
7aad4be8a48e print values for stack entry;
wenzelm
parents: 60896
diff changeset
   284
65213
51c0f094dc02 proper local debugger state, depending on session;
wenzelm
parents: 63805
diff changeset
   285
  def print_vals(c: Debugger.Context, SML: Boolean, context: String)
60897
7aad4be8a48e print values for stack entry;
wenzelm
parents: 60896
diff changeset
   286
  {
61014
39f67bb4e609 maintain per-thread focus context;
wenzelm
parents: 61011
diff changeset
   287
    require(c.debug_index.isDefined)
61010
cccfd7f6317d more explicit type Debugger.Context;
wenzelm
parents: 61008
diff changeset
   288
65213
51c0f094dc02 proper local debugger state, depending on session;
wenzelm
parents: 63805
diff changeset
   289
    state.change(st => {
61014
39f67bb4e609 maintain per-thread focus context;
wenzelm
parents: 61011
diff changeset
   290
      input(c.thread_name, "print_vals", c.debug_index.getOrElse(0).toString,
61010
cccfd7f6317d more explicit type Debugger.Context;
wenzelm
parents: 61008
diff changeset
   291
        SML.toString, Symbol.encode(context))
65213
51c0f094dc02 proper local debugger state, depending on session;
wenzelm
parents: 63805
diff changeset
   292
      st.clear_output(c.thread_name)
60897
7aad4be8a48e print values for stack entry;
wenzelm
parents: 60896
diff changeset
   293
    })
60902
9f185acfdcb8 clarified events;
wenzelm
parents: 60901
diff changeset
   294
    delay_update.invoke()
60897
7aad4be8a48e print values for stack entry;
wenzelm
parents: 60896
diff changeset
   295
  }
60749
f727b99faaf7 skeleton for interactive debugger;
wenzelm
parents:
diff changeset
   296
}