view libinterp/parse-tree/pt-eval.h @ 33571:742d8fc77688 default tip @

Support setting breakpoints in get and set methods of classdef properties (bug #65610). * cdef-class.cc (cdef_class::cdef_class_rep::get_method): Also check for any `get` or `set` methods of `classdef` properties. * bp-table.cc (user_code_provider::operator ()): Support getting (closest) user code to `get` or `set` methods of `classdef` classes. (user_code_provider::populate_function_cache): Add `get` and `set` methods to function cache for `classdef` classes. * pt-eval.cc (tree_evaluator::get_user_code): Support getting user code for `get` or `set` methods of `classdef` properties. * test/classdef-debug/classdef_breakpoints2.m: Add handle class with get and set methods for new self tests. * test/classdef-debug/test-classdef-breakpoints.tst: Add new tests for adding and clearing breakpoints in `set` and `get` methods of `classdef` properties by line number or function name. Make sure breakpoints are deleted in existing tests also on test failures. Fix syntax error in 69eb4c27d8c8. * test/classdef-debug/module.mk: Add new file to build system. * etc/NEWS.10.md: Add note about new feature.
author Markus Mützel <markus.muetzel@gmx.de>
date Sat, 20 Apr 2024 13:13:50 +0200
parents 50e19b01e111
children 3bb4422bd982
line wrap: on
line source

////////////////////////////////////////////////////////////////////////
//
// Copyright (C) 2009-2024 The Octave Project Developers
//
// See the file COPYRIGHT.md in the top-level directory of this
// distribution or <https://octave.org/copyright/>.
//
// This file is part of Octave.
//
// Octave is free software: you can redistribute it and/or modify it
// under the terms of the GNU General Public License as published by
// the Free Software Foundation, either version 3 of the License, or
// (at your option) any later version.
//
// Octave is distributed in the hope that it will be useful, but
// WITHOUT ANY WARRANTY; without even the implied warranty of
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
// GNU General Public License for more details.
//
// You should have received a copy of the GNU General Public License
// along with Octave; see the file COPYING.  If not, see
// <https://www.gnu.org/licenses/>.
//
////////////////////////////////////////////////////////////////////////

#if ! defined (octave_pt_eval_h)
#define octave_pt_eval_h 1

#include "octave-config.h"

#include <iosfwd>
#include <list>
#include <memory>
#include <set>
#include <stack>
#include <string>

#include "bp-table.h"
#include "call-stack.h"
#include "oct-lvalue.h"
#include "ov.h"
#include "ovl.h"
#include "profiler.h"
#include "pt-walk.h"
#include "stack-frame.h"

class octave_builtin;
class octave_mex_function;
class octave_user_code;
class octave_user_function;
class octave_user_script;

OCTAVE_BEGIN_NAMESPACE(octave)

class symbol_info_list;
class symbol_scope;
class tree_decl_elt;
class tree_expression;

class debugger;
class interpreter;
class push_parser;
class unwind_protect;

// How to evaluate the code that the parse trees represent.

class OCTINTERP_API tree_evaluator : public tree_walker
{
public:

  enum echo_state
  {
    ECHO_OFF = 0,
    ECHO_SCRIPTS = 1,
    ECHO_FUNCTIONS = 2,
    ECHO_ALL = 4
  };

  template <typename T>
  class value_stack
  {
  public:

    OCTAVE_DEFAULT_CONSTRUCT_COPY_MOVE_DELETE (value_stack)

    void push (const T& val) { m_stack.push (val); }

    void pop ()
    {
      m_stack.pop ();
    }

    T val_pop ()
    {
      T retval = m_stack.top ();
      m_stack.pop ();
      return retval;
    }

    T top () const
    {
      return m_stack.top ();
    }

    std::size_t size () const
    {
      return m_stack.size ();
    }

    bool empty () const
    {
      return m_stack.empty ();
    }

    void clear ()
    {
      while (! m_stack.empty ())
        m_stack.pop ();
    }

  private:

    std::stack<T> m_stack;
  };

  typedef void (*decl_elt_init_fcn) (tree_decl_elt&);

  tree_evaluator (interpreter& interp)
    : m_interpreter (interp), m_parser (), m_statement_context (SC_OTHER),
      m_lvalue_list (nullptr), m_autoload_map (), m_bp_table (*this),
      m_call_stack (*this), m_profiler (), m_debug_frame (0),
      m_debug_mode (false), m_quiet_breakpoint_flag (false),
      m_debugger_stack (), m_exit_status (0), m_max_recursion_depth (256),
      m_whos_line_format ("  %la:5; %ln:6; %cs:16:6:1;  %rb:12;  %lc:-1;\n"),
      m_silent_functions (false), m_string_fill_char (' '), m_PS4 ("+ "),
      m_dbstep_flag (0), m_break_on_next_stmt (false), m_echo (ECHO_OFF),
      m_echo_state (false), m_echo_file_name (),
      m_echo_file_pos (1),
      m_echo_files (), m_in_top_level_repl (false),
      m_server_mode (false), m_in_loop_command (false),
      m_breaking (0), m_continuing (0), m_returning (0),
      m_indexed_object (), m_index_list (), m_index_type (),
      m_index_position (0), m_num_indices (0)
  { }

  OCTAVE_DISABLE_CONSTRUCT_COPY_MOVE (tree_evaluator)

  ~tree_evaluator () = default;

  std::shared_ptr<push_parser> get_parser ()
  {
    return m_parser;
  }

  void set_parser (const std::shared_ptr<push_parser>& parser)
  {
    m_parser = parser;
  }

  bool at_top_level () const;

  std::string mfilename (const std::string& opt = "") const;

  // Parse a line of input.  If input ends at a complete statement
  // boundary, execute the resulting parse tree.  Useful to handle
  // parsing user input when running in server mode.

  void parse_and_execute (const std::string& input, bool& incomplete_parse);

  void get_line_and_eval ();

  int repl ();

  bool in_top_level_repl () const { return m_in_top_level_repl; }

  int server_loop ();

  bool server_mode () const { return m_server_mode; }

  void server_mode (bool arg) { m_server_mode = arg; }

  void eval (std::shared_ptr<tree_statement_list>& stmt_list,
             bool interactive);

  octave_value_list eval_string (const std::string& eval_str, bool silent,
                                 int& parse_status, int nargout);

  octave_value eval_string (const std::string& eval_str, bool silent,
                            int& parse_status);

  octave_value_list eval_string (const octave_value& arg, bool silent,
                                 int& parse_status, int nargout);

  octave_value_list eval (const std::string& try_code, int nargout);

  octave_value_list eval (const std::string& try_code,
                          const std::string& catch_code, int nargout);

  octave_value_list evalin (const std::string& context,
                            const std::string& try_code, int nargout);

  octave_value_list evalin (const std::string& context,
                            const std::string& try_code,
                            const std::string& catch_code, int nargout);

  void visit_anon_fcn_handle (tree_anon_fcn_handle&);

  void visit_argument_list (tree_argument_list&);

  void visit_arguments_block (tree_arguments_block&);

  void visit_args_block_attribute_list (tree_args_block_attribute_list&);

  void visit_args_block_validation_list (tree_args_block_validation_list&);

  void visit_arg_validation (tree_arg_validation&);

  void visit_arg_size_spec (tree_arg_size_spec&);

  void visit_arg_validation_fcns (tree_arg_validation_fcns&);

  void visit_binary_expression (tree_binary_expression&);

  void visit_boolean_expression (tree_boolean_expression&);

  void visit_compound_binary_expression (tree_compound_binary_expression&);

  void visit_break_command (tree_break_command&);

  void visit_colon_expression (tree_colon_expression&);

  void visit_continue_command (tree_continue_command&);

  void visit_decl_command (tree_decl_command&);

  void visit_decl_elt (tree_decl_elt&);

  void visit_simple_for_command (tree_simple_for_command&);

  void visit_complex_for_command (tree_complex_for_command&);

  void visit_spmd_command (tree_spmd_command&);

  octave_value evaluate_anon_fcn_handle (tree_anon_fcn_handle& afh);

  octave_value_list
  execute_builtin_function (octave_builtin& builtin_function, int nargout,
                            const octave_value_list& args);
  octave_value_list
  execute_mex_function (octave_mex_function& mex_function, int nargout,
                        const octave_value_list& args);

  void visit_octave_user_script (octave_user_script&);

  octave_value_list
  execute_user_script (octave_user_script& user_script, int nargout,
                       const octave_value_list& args);

  void visit_octave_user_function (octave_user_function&);

  octave_value_list
  execute_user_function (octave_user_function& user_function,
                         int nargout, const octave_value_list& args);

  void visit_octave_user_function_header (octave_user_function&);

  void visit_octave_user_function_trailer (octave_user_function&);

  void visit_function_def (tree_function_def&);

  void visit_identifier (tree_identifier&);

  void visit_if_clause (tree_if_clause&);

  void visit_if_command (tree_if_command&);

  void visit_if_command_list (tree_if_command_list&);

  void visit_index_expression (tree_index_expression&);

  void visit_matrix (tree_matrix&);

  void visit_cell (tree_cell&);

  void visit_multi_assignment (tree_multi_assignment&);

  void visit_no_op_command (tree_no_op_command&);

  void visit_constant (tree_constant&);

  void visit_fcn_handle (tree_fcn_handle&);

  void visit_parameter_list (tree_parameter_list&);

  void visit_postfix_expression (tree_postfix_expression&);

  void visit_prefix_expression (tree_prefix_expression&);

  void visit_return_command (tree_return_command&);

  void visit_simple_assignment (tree_simple_assignment&);

  void visit_statement (tree_statement&);

  void visit_statement_list (tree_statement_list&);

  void visit_switch_case (tree_switch_case&);

  void visit_switch_case_list (tree_switch_case_list&);

  void visit_switch_command (tree_switch_command&);

  void visit_try_catch_command (tree_try_catch_command&);

  void do_unwind_protect_cleanup_code (tree_statement_list *list);

  void visit_unwind_protect_command (tree_unwind_protect_command&);

  void visit_while_command (tree_while_command&);
  void visit_do_until_command (tree_do_until_command&);

  void visit_superclass_ref (tree_superclass_ref&);
  void visit_metaclass_query (tree_metaclass_query&);

  void bind_ans (const octave_value& val, bool print);

  bool statement_printing_enabled ();

  void reset_debug_state ();

  void reset_debug_state (bool mode);

  void enter_debugger (const std::string& prompt = "debug> ");

  void keyboard (const std::string& prompt = "keyboard> ");

  void dbupdown (int n, bool verbose = false);

  // Possible types of evaluation contexts.
  enum stmt_list_type
  {
    SC_FUNCTION,  // function body
    SC_SCRIPT,    // script file
    SC_OTHER      // command-line input or eval string
  };

  Matrix ignored_fcn_outputs () const;

  std::string inputname (int n, bool ids_only = true) const;

  octave_value make_fcn_handle (const std::string& nm);

  octave_value evaluate (tree_decl_elt *);

  void install_variable (const std::string& name,
                         const octave_value& value, bool global);

  octave_value global_varval (const std::string& name) const;

  octave_value& global_varref (const std::string& name);

  void global_assign (const std::string& name,
                      const octave_value& val = octave_value ());

  octave_value top_level_varval (const std::string& name) const;

  void top_level_assign (const std::string& name,
                         const octave_value& val = octave_value ());

  bool is_variable (const std::string& name) const;

  bool is_local_variable (const std::string& name) const;

  bool is_variable (const tree_expression *expr) const;

  bool is_defined (const tree_expression *expr) const;

  bool is_variable (const symbol_record& sym) const;

  bool is_defined (const symbol_record& sym) const;

  bool is_global (const std::string& name) const;

  octave_value varval (const symbol_record& sym) const;

  octave_value varval (const std::string& name) const;

  void assign (const std::string& name,
               const octave_value& val = octave_value ());

  void assignin (const std::string& context, const std::string& name,
                 const octave_value& val = octave_value ());

  void source_file (const std::string& file_name,
                    const std::string& context = "",
                    bool verbose = false, bool require_file = true);

  void set_auto_fcn_var (stack_frame::auto_var_type avt,
                         const octave_value& val = octave_value ());

  void set_nargin (int nargin);
  void set_nargout (int nargout);

  octave_value get_auto_fcn_var (stack_frame::auto_var_type avt) const;

  void define_parameter_list_from_arg_vector
  (tree_parameter_list *param_list, const octave_value_list& args);

  void undefine_parameter_list (tree_parameter_list *param_list);

  octave_value_list convert_to_const_vector (tree_argument_list *arg_list);

  octave_value_list
  convert_return_list_to_const_vector
  (tree_parameter_list *ret_list, int nargout, const Cell& varargout);

  bool eval_decl_elt (tree_decl_elt *elt);

  bool switch_case_label_matches (tree_switch_case *expr,
                                  const octave_value& val);

  interpreter& get_interpreter () { return m_interpreter; }

  bp_table& get_bp_table () { return m_bp_table; }

  profiler& get_profiler () { return m_profiler; }

  void push_stack_frame (const symbol_scope& scope);

  void push_stack_frame (octave_user_function *fcn,
                         const std::shared_ptr<stack_frame>& closure_frames = std::shared_ptr<stack_frame> ());

  void push_stack_frame (octave_user_function *fcn,
                         const stack_frame::local_vars_map& local_vars,
                         const std::shared_ptr<stack_frame>& closure_frames = std::shared_ptr<stack_frame> ());

  void push_stack_frame (octave_user_script *script);

  void push_stack_frame (octave_function *fcn);

  void pop_stack_frame ();

  std::shared_ptr<stack_frame> pop_return_stack_frame ();

  std::shared_ptr<stack_frame> get_current_stack_frame () const
  {
    return m_call_stack.get_current_stack_frame ();
  }

  std::shared_ptr<stack_frame> current_user_frame () const
  {
    return m_call_stack.current_user_frame ();
  }

  // Current line in current function.
  int current_line () const;

  // Current column in current function.
  int current_column () const;

  // Line number in current function that we are debugging.
  int debug_user_code_line () const;

  // Column number in current function that we are debugging.
  int debug_user_code_column () const;

  void debug_where (std::ostream& os) const;

  void debug_list (std::ostream& os, int num_lines) const;

  void debug_type (std::ostream& os, int start_line, int end_line) const;

  octave_user_code * current_user_code () const;

  unwind_protect * curr_fcn_unwind_protect_frame ();

  // Current function that we are debugging.
  octave_user_code * debug_user_code () const;

  octave_function * current_function (bool skip_first = false) const;

  octave_function * caller_function () const;

  bool goto_frame (std::size_t n = 0, bool verbose = false);

  void goto_caller_frame ();

  void goto_base_frame ();

  void restore_frame (std::size_t n);

  std::string get_dispatch_class () const;

  void set_dispatch_class (const std::string& class_name);

  bool is_class_method_executing (std::string& dispatch_class) const;

  bool is_class_constructor_executing (std::string& dispatch_class) const;

  std::list<std::shared_ptr<stack_frame>>
                                       backtrace_frames (octave_idx_type& curr_user_frame) const;

  std::list<std::shared_ptr<stack_frame>> backtrace_frames () const;

  std::list<frame_info> backtrace_info (octave_idx_type& curr_user_frame,
                                        bool print_subfn = true) const;

  std::list<frame_info> backtrace_info () const;

  octave_map backtrace (octave_idx_type& curr_user_frame,
                        bool print_subfn = true) const;

  octave_map backtrace () const;

  octave_map empty_backtrace () const;

  std::string backtrace_message () const;

  void push_dummy_scope (const std::string& name);
  void pop_scope ();

  symbol_scope get_top_scope () const;
  symbol_scope get_current_scope () const;

  void mlock (bool skip_first = false) const;

  void munlock (bool skip_first = false) const;

  bool mislocked (bool skip_first = false) const;

  octave_value max_stack_depth (const octave_value_list& args, int nargout);

  // Useful for debugging
  void display_call_stack () const;

  octave_value find (const std::string& name);

  void clear_objects ();

  void clear_variable (const std::string& name);

  void clear_variable_pattern (const std::string& pattern);

  void clear_variable_regexp (const std::string& pattern);

  void clear_variables ();

  void clear_global_variable (const std::string& name);

  void clear_global_variable_pattern (const std::string& pattern);

  void clear_global_variable_regexp (const std::string& pattern);

  void clear_global_variables ();

  void clear_all (bool force = false);

  void clear_symbol (const std::string& name);

  void clear_symbol_pattern (const std::string& pattern);

  void clear_symbol_regexp (const std::string& pattern);

  std::list<std::string> global_variable_names () const;

  std::list<std::string> top_level_variable_names () const;

  std::list<std::string> variable_names () const;

  octave_user_code * get_user_code (const std::string& fname = "");

  std::string current_function_name (bool skip_first = false) const;

  bool in_user_code () const;

  symbol_info_list glob_symbol_info (const std::string& pattern) const;

  symbol_info_list regexp_symbol_info (const std::string& pattern) const;

  symbol_info_list get_symbol_info ();

  symbol_info_list top_scope_symbol_info () const;

  octave_map get_autoload_map () const;

  std::string lookup_autoload (const std::string& nm) const;

  std::list<std::string> autoloaded_functions () const;

  std::list<std::string> reverse_lookup_autoload (const std::string& nm) const;

  void add_autoload (const std::string& fcn, const std::string& nm);

  void remove_autoload (const std::string& fcn, const std::string& nm);

  int max_recursion_depth () const { return m_max_recursion_depth; }

  int max_recursion_depth (int n)
  {
    int val = m_max_recursion_depth;
    m_max_recursion_depth = n;
    return val;
  }

  octave_value
  max_recursion_depth (const octave_value_list& args, int nargout);

  bool silent_functions () const { return m_silent_functions; }

  bool silent_functions (bool b)
  {
    int val = m_silent_functions;
    m_silent_functions = b;
    return val;
  }

  octave_value whos_line_format (const octave_value_list& args, int nargout);

  std::string whos_line_format () const { return m_whos_line_format; }

  std::string whos_line_format (const std::string& s)
  {
    std::string val = m_whos_line_format;
    m_whos_line_format = s;
    return val;
  }

  octave_value
  silent_functions (const octave_value_list& args, int nargout);

  std::size_t debug_frame () const { return m_debug_frame; }

  std::size_t debug_frame (std::size_t n)
  {
    std::size_t val = m_debug_frame;
    m_debug_frame = n;
    return val;
  }

  std::size_t current_call_stack_frame_number () const
  {
    return m_call_stack.current_frame ();
  }

  bool quiet_breakpoint_flag () const { return m_quiet_breakpoint_flag; }

  bool quiet_breakpoint_flag (bool flag)
  {
    bool val = m_quiet_breakpoint_flag;
    m_quiet_breakpoint_flag = flag;
    return val;
  }

  char string_fill_char () const { return m_string_fill_char; }

  char string_fill_char (char c)
  {
    int val = m_string_fill_char;
    m_string_fill_char = c;
    return val;
  }

  // The following functions are provided for convenience.  They
  // call the corresponding functions in the debugger class for the
  // current debugger (if any).

  bool in_debug_repl () const;

  void dbcont ();

  // Return true if we are in the debug repl and m_execution_mode is
  // set to exit the debugger.  Otherwise, do nothing.

  void dbquit (bool all = false);

  // Add EXPR to the set of expressions that may be evaluated when the
  // debugger stops at a breakpoint.
  void add_debug_watch_expression (const std::string& expr)
  {
    m_debug_watch_expressions.insert (expr);
  }

  // Remove EXPR from the set of expressions that may be evaluated
  // when the debugger stops at a breakpoint.
  void remove_debug_watch_expression (const std::string& expr)
  {
    m_debug_watch_expressions.erase (expr);
  }

  // Clear the set of expressions that may be evaluated when the
  // debugger stops at a breakpoint.
  void clear_debug_watch_expressions ()
  {
    m_debug_watch_expressions.clear ();
  }

  // Return the set of expressions that may be evaluated when the
  // debugger stops at a breakpoint.
  std::set<std::string> debug_watch_expressions () const
  {
    return m_debug_watch_expressions;
  }

  octave_value PS4 (const octave_value_list& args, int nargout);

  std::string PS4 () const { return m_PS4; }

  std::string PS4 (const std::string& s)
  {
    std::string val = m_PS4;
    m_PS4 = s;
    return val;
  }

  void set_PS4 (const std::string& s) { m_PS4 = s; }

  octave_value indexed_object () const
  {
    return m_indexed_object;
  }

  void set_indexed_object (const octave_value& obj = octave_value ())
  {
    m_indexed_object = obj;
  }

  const std::list<octave_value_list>& index_list () const
  {
    return m_index_list;
  }

  void set_index_list (const std::string& index_type,
                       const std::list<octave_value_list>& index_list)
  {
    m_index_type = index_type;
    m_index_list = index_list;
  }

  void clear_index_list ()
  {
    m_index_type = "";
    m_index_list.clear ();
  }

  void append_index_list (char type, const octave_value_list& idx)
  {
    m_index_type += type;
    m_index_list.push_back (idx);
  }

  const std::string& index_type () const
  {
    return m_index_type;
  }

  int index_position () const { return m_index_position; }

  int num_indices () const { return m_num_indices; }

  octave_value_list evaluate_end_expression (const octave_value_list& args);

  const std::list<octave_lvalue> * lvalue_list () const
  {
    return m_lvalue_list;
  }

  void set_lvalue_list (const std::list<octave_lvalue> *lst)
  {
    m_lvalue_list = lst;
  }

  int breaking () const { return m_breaking; }

  int breaking (int n)
  {
    int val = m_breaking;
    m_breaking = n;
    return val;
  }

  int continuing () const { return m_continuing; }

  int continuing (int n)
  {
    int val = m_continuing;
    m_continuing = n;
    return val;
  }

  int returning () const { return m_returning; }

  int returning (int n)
  {
    int val = m_returning;
    m_returning = n;
    return val;
  }

  int dbstep_flag () const { return m_dbstep_flag; }

  int dbstep_flag (int val)
  {
    int old_val = m_dbstep_flag;
    m_dbstep_flag = val;
    return old_val;
  }

  void set_dbstep_flag (int step) { m_dbstep_flag = step; }

  bool break_on_next_statement () const
  {
    return m_break_on_next_stmt;
  }

  bool break_on_next_statement (bool val)
  {
    bool old_val = m_break_on_next_stmt;
    m_break_on_next_stmt = val;
    return old_val;
  }

  void set_break_on_next_statement (bool val)
  {
    m_break_on_next_stmt = val;
  }

  octave_value echo (const octave_value_list& args, int nargout);

  int echo () const { return m_echo; }

  int echo (int val)
  {
    int old_val = m_echo;
    m_echo = val;

    return old_val;
  }

  octave_value
  string_fill_char (const octave_value_list& args, int nargout);

  void final_index_error (index_exception& ie, const tree_expression *expr);

  octave_value do_who (int argc, const string_vector& argv,
                       bool return_list, bool verbose = false);

  octave_value_list
  make_value_list (tree_argument_list *args, const string_vector& arg_nm);

  std::list<octave_lvalue> make_lvalue_list (tree_argument_list *);

  void push_echo_state (int type, const std::string& file_name, int pos = 1);

  bool debug_mode () const { return m_debug_mode; }

  int echo_state () { return m_echo_state; }

  void set_echo_file_pos (int pos)
  {
    m_echo_file_pos = pos;
  }

private:

  template <typename T>
  void execute_range_loop (const range<T>& rng, int line,
                           octave_lvalue& ult,
                           tree_statement_list *loop_body);

  void set_echo_state (int type, const std::string& file_name, int pos);

  void maybe_set_echo_state ();

  void push_echo_state_cleanup (unwind_protect& frame);

  bool maybe_push_echo_state_cleanup ();

  void do_breakpoint (tree_statement& stmt);

  void do_breakpoint (bool is_breakpoint,
                      bool is_end_of_fcn_or_script = false);

  bool is_logically_true (tree_expression *expr, const char *warn_for);

  // For unwind-protect.
  void uwp_set_echo_state (bool state, const std::string& file_name, int pos);

  bool echo_this_file (const std::string& file, int type) const;

  void echo_code (int line);

  bool quit_loop_now ();

  void bind_auto_fcn_vars (const string_vector& arg_names,
                           const Matrix& ignored_outputs, int nargin,
                           int nargout, bool takes_varargs,
                           const octave_value_list& va_args);

  std::string check_autoload_file (const std::string& nm) const;

  interpreter& m_interpreter;

  std::shared_ptr<push_parser> m_parser;

  // The context for the current evaluation.
  stmt_list_type m_statement_context;

  const std::list<octave_lvalue> *m_lvalue_list;

  // List of autoloads (function -> file mapping).
  std::map<std::string, std::string> m_autoload_map;

  bp_table m_bp_table;

  call_stack m_call_stack;

  profiler m_profiler;

  // The number of the stack frame we are currently debugging.
  std::size_t m_debug_frame;

  bool m_debug_mode;

  bool m_quiet_breakpoint_flag;

  // When entering the debugger we push it on this stack.  Managing
  // debugger invocations this way allows us to handle recursive
  // debugger calls.  When we exit a debugger the object is popped
  // from the stack and deleted and we resume working with the
  // previous debugger (if any) that is now at the top of the stack.
  std::stack<debugger *> m_debugger_stack;

  std::set<std::string> m_debug_watch_expressions;

  int m_exit_status;

  // Maximum nesting level for functions, scripts, or sourced files
  // called recursively.
  int m_max_recursion_depth;

  // Defines layout for the whos/who -long command
  std::string m_whos_line_format;

  // If TRUE, turn off printing of results in functions (as if a
  // semicolon has been appended to each statement).
  bool m_silent_functions;

  // The character to fill with when creating string arrays.
  char m_string_fill_char;

  // String printed before echoed commands (enabled by --echo-commands).
  std::string m_PS4;

  // If > 0, stop executing at the (N-1)th stopping point, counting
  //         from the the current execution point in the current frame.
  //
  // If < 0, stop executing at the next possible stopping point.
  int m_dbstep_flag;

  // If TRUE, and we are not stopping for another reason (dbstep or a
  // breakpoint) then stop at next statement and enter the debugger.
  bool m_break_on_next_stmt;

  // Echo commands as they are executed?
  //
  //   1 => echo commands read from script files
  //   2 => echo commands from functions
  //
  // more than one state can be active at once.
  int m_echo;

  // Are we currently echoing commands?  This state is set by the
  // functions that execute functions and scripts.
  bool m_echo_state;

  std::string m_echo_file_name;

  // Next line to echo, counting from 1.  We use int here because the
  // parser does.  It also initializes line and column numbers to the
  // invalid value -1 and that can cause trouble if cast to an
  // unsigned value.  When updating this value and echoing ranges of
  // code, we also check to ensure that the line numbers stored in the
  // parse tree are valid.  It would be better to ensure that the
  // parser always stores valid position info, but that's more
  // difficult to always do correctly.
  int m_echo_file_pos;

  std::map<std::string, bool> m_echo_files;

  // TRUE if we are in the top level interactive read eval print loop.
  bool m_in_top_level_repl;

  // TRUE means we are executing in the server_loop function.
  bool m_server_mode;

  // TRUE means we are evaluating some kind of looping construct.
  bool m_in_loop_command;

  // Nonzero means we're breaking out of a loop or function body.
  int m_breaking;

  // Nonzero means we're jumping to the end of a loop.
  int m_continuing;

  // Nonzero means we're returning from a function.
  int m_returning;

  // The following are all used by the END function.  Maybe they
  // should be kept together in a separate object?
  octave_value m_indexed_object;
  std::list<octave_value_list> m_index_list;
  std::string m_index_type;
  int m_index_position;
  int m_num_indices;
};

OCTAVE_END_NAMESPACE(octave)

#endif