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