[nemiver] ASM <-> Source code switching preliminary support
- From: Dodji Seketeli <dodji src gnome org>
- To: commits-list gnome org
- Cc:
- Subject: [nemiver] ASM <-> Source code switching preliminary support
- Date: Sat, 17 Jul 2010 21:32:02 +0000 (UTC)
commit e6f0436332cf8f5426d92291a1bcfee54df829a2
Author: Dodji Seketeli <dodji redhat com>
Date: Sun Mar 7 21:02:08 2010 +0100
ASM <-> Source code switching preliminary support
Preliminary support to switch from source code view to assembly
view and backward. Neiher decorating nor stepping is supported
in the assembly view yet.
* src/uicommon/nmv-source-editor.h (clear_decorations): New
interface.
(SourceEditor::*): Renamed all methods that had "composite" in
their names into method that have "assembly" in their name
instead.
* src/uicommon/nmv-source-editor.cc (*):
Same similar renaming as above.
Go further by renaming members that have "comp" in their names
by members that have "asm" in their names instead.
(SourceEditor::clear_decorations): New definition.
* src/persp/dbgperspective/nmv-dbg-perspective.cc
(DBGPerspective::on_disassemble_action,
DBGPerspective::on_switch_to_disassembly_action,
DBGPerspective::on_debugger_disassembled_signal,
DBGPerspective::switch_to_disassembly,
DBGPerspective::switch_to_source_code,
DBGPerspective::disassemble,
DBGPerspective::find_absolute_path_or_ask_user,
DBGPerspective::remove_visual_decorations_from_text): New fns
declarations and definitions.
(DBGPerspective::find_absolute_path,
DBGPerspective::ask_user_to_select_file): Extract these
functions from DBGPerspective::append_visual_breakpoint.
(DBGPerspective::append_visual_breakpoint): Use the new
DBGPerspective::find_absolute_path and
DBGPerspective::ask_user_to_select_file.
(DBGPerspective::init_actions): Add menu actions DisAsmMenuItemAction,
SwitchToAsmMenuItemAction, SwitchToSourceMenuItemAction. Used
the new DBGPerspective::on_disassemble_action, and
DBGPerspective::switch_to_source_code as the callbacks for this
menu actions.
(DBGPerspective::open_disassembly): If the disassembly tab
exists already, re-use it.
(DBGPerspective::open_file): Don't yell if the file path is empty.
* src/persp/dbgperspective/menus/menus.xml: Add new "Show assembly",
"Switch to assembly" and "Switch to source" menu items to the
"debug" menu.
src/persp/dbgperspective/menus/menus.xml | 6 +
src/persp/dbgperspective/nmv-dbg-perspective.cc | 440 +++++++++++++++++------
src/uicommon/nmv-source-editor.cc | 180 ++++++----
src/uicommon/nmv-source-editor.h | 29 +-
4 files changed, 455 insertions(+), 200 deletions(-)
---
diff --git a/src/persp/dbgperspective/menus/menus.xml b/src/persp/dbgperspective/menus/menus.xml
index 75967c5..ccdb2f1 100644
--- a/src/persp/dbgperspective/menus/menus.xml
+++ b/src/persp/dbgperspective/menus/menus.xml
@@ -85,6 +85,12 @@
name="ActivateGlobalVariablesDialogMenuItem"/>
<menuitem action="RefreshLocalVariablesMenuItemAction"
name="RefreshLocalVariablesMenuItemAction"/>
+ <menuitem action="DisAsmMenuItemAction"
+ name="DisAsmMenuItemAction"/>
+ <menuitem action="SwitchToAsmMenuItemAction"
+ name="SwitchToAsmMenuItemAction"/>
+ <menuitem action="SwitchToSourceMenuItemAction"
+ name="SwitchToSourceMenuItemAction"/>
</menu>
</placeholder>
</menubar>
diff --git a/src/persp/dbgperspective/nmv-dbg-perspective.cc b/src/persp/dbgperspective/nmv-dbg-perspective.cc
index 30faac7..975afb5 100644
--- a/src/persp/dbgperspective/nmv-dbg-perspective.cc
+++ b/src/persp/dbgperspective/nmv-dbg-perspective.cc
@@ -132,6 +132,8 @@ const char *LAST_RUN_TIME= "lastruntime";
const char *DBG_PERSPECTIVE_MOUSE_MOTION_DOMAIN =
"dbg-perspective-mouse-motion-domain";
+static const int NUM_INSTR_TO_DISASSEMBLE = 20;
+
const char* CONF_KEY_NEMIVER_SOURCE_DIRS =
"/apps/nemiver/dbgperspective/source-search-dirs";
const char* CONF_KEY_SHOW_DBG_ERROR_DIALOGS =
@@ -279,6 +281,8 @@ private:
void on_set_breakpoint_using_dialog_action ();
void on_set_watchpoint_using_dialog_action ();
void on_refresh_locals_action ();
+ void on_disassemble_action (bool a_show_asm_in_new_tab);
+ void on_switch_to_disassembly_action ();
void on_toggle_breakpoint_action ();
void on_toggle_breakpoint_enabled_action ();
void on_inspect_variable_action ();
@@ -321,6 +325,7 @@ private:
void on_insertion_changed_signal (const Gtk::TextBuffer::iterator& iter,
SourceEditor *a_editor);
+
void update_toggle_menu_text (const UString& a_current_file,
int a_current_line);
@@ -340,7 +345,7 @@ private:
void on_debugger_detached_from_target_signal ();
void on_debugger_got_target_info_signal (int a_pid,
- const UString &a_exe_path);
+ const UString &a_exe_path);
void on_debugger_console_message_signal (const UString &a_msg);
@@ -387,6 +392,10 @@ private:
(const UString &a_var_name,
const IDebugger::VariableSafePtr &a_var,
const UString &a_cooker);
+ void on_debugger_disassembled_signal
+ (const IDebugger::DisassembleInfo &a_info,
+ const std::list<IDebugger::AsmInstr> &a_instrs,
+ bool a_show_asm_in_new_tab = true);
void on_variable_created_for_tooltip_signal
(const IDebugger::VariableSafePtr);
@@ -523,6 +532,11 @@ public:
bool open_disassembly (const IDebugger::DisassembleInfo &a_info,
const std::list<IDebugger::AsmInstr> &a_asm);
+ void switch_to_disassembly (const IDebugger::DisassembleInfo &a_info,
+ const std::list<IDebugger::AsmInstr> &a_asm);
+
+ void switch_to_source_code ();
+
void close_opened_files ();
void update_file_maps ();
@@ -606,6 +620,7 @@ public:
void set_breakpoint_from_dialog (SetBreakpointDialog &a_dialog);
void set_watchpoint_using_dialog ();
void refresh_locals ();
+ void disassemble (bool a_show_asm_in_new_tab);
void inspect_variable ();
void inspect_variable (const UString &a_variable_name);
@@ -615,7 +630,15 @@ public:
void toggle_breakpoint_enabled (const UString &a_file_path,
int a_linenum);
void toggle_breakpoint_enabled ();
+
void update_src_dependant_bp_actions_sensitiveness ();
+
+ bool find_absolute_path (const UString& a_file_path,
+ UString& a_absolute_path);
+ bool find_absolute_path_or_ask_user (const UString& a_file_path,
+ UString& a_absolute_path);
+ bool ask_user_to_select_file (const UString &a_file_name,
+ UString& a_selected_file_path);
bool append_visual_breakpoint (const UString &a_file_name,
int a_linenum,
UString &a_actual_file_name,
@@ -628,6 +651,7 @@ public:
void choose_function_overload
(const vector<IDebugger::OverloadsChoiceEntry> &a_entries);
+ void remove_visual_decorations_from_text (const UString &a_file_path);
bool apply_decorations_to_text (const UString &a_file_path);
IDebuggerSafePtr& debugger ();
@@ -1496,6 +1520,23 @@ DBGPerspective::on_refresh_locals_action ()
}
void
+DBGPerspective::on_disassemble_action (bool a_show_asm_in_new_tab)
+{
+ LOG_FUNCTION_SCOPE_NORMAL_DD;
+ NEMIVER_TRY
+ disassemble (a_show_asm_in_new_tab);
+ NEMIVER_CATCH
+}
+
+void
+DBGPerspective::on_switch_to_disassembly_action ()
+{
+ LOG_FUNCTION_SCOPE_NORMAL_DD;
+ NEMIVER_TRY
+ NEMIVER_CATCH
+}
+
+void
DBGPerspective::on_toggle_breakpoint_enabled_action ()
{
LOG_FUNCTION_SCOPE_NORMAL_DD;
@@ -2572,6 +2613,20 @@ DBGPerspective::on_debugger_variable_value_signal
}
void
+DBGPerspective::on_debugger_disassembled_signal
+ (const IDebugger::DisassembleInfo &a_info,
+ const std::list<IDebugger::AsmInstr> &a_instrs,
+ bool a_show_asm_in_new_tab)
+{
+ LOG_FUNCTION_SCOPE_NORMAL_DD;
+
+ if (a_show_asm_in_new_tab)
+ open_disassembly (a_info, a_instrs);
+ else
+ switch_to_disassembly (a_info, a_instrs);
+}
+
+void
DBGPerspective::on_variable_created_for_tooltip_signal
(const IDebugger::VariableSafePtr a_var)
{
@@ -3197,7 +3252,43 @@ DBGPerspective::init_actions ()
ActionEntry::DEFAULT,
"",
false
- }
+ },
+ {
+ "DisAsmMenuItemAction",
+ nil_stock_id,
+ _("Show assembly"),
+ _("Show the assembly code of the source code being "
+ "currently debugged, in another tab"),
+ sigc::bind (sigc::mem_fun
+ (*this, &DBGPerspective::on_disassemble_action),
+ /*show_asm_in_new_tab=*/true),
+ ActionEntry::DEFAULT,
+ "",
+ false
+ },
+ {
+ "SwitchToAsmMenuItemAction",
+ nil_stock_id,
+ _("Switch to assembly"),
+ _("Show the assembly code of the source code being "
+ "currently debugged"),
+ sigc::bind (sigc::mem_fun
+ (*this, &DBGPerspective::on_disassemble_action),
+ /*show_asm_in_new_tab=*/false),
+ ActionEntry::DEFAULT,
+ "",
+ false
+ },
+ {
+ "SwitchToSourceMenuItemAction",
+ nil_stock_id,
+ _("Switch to source"),
+ _("Show the source code being currently debugged"),
+ sigc::mem_fun (*this, &DBGPerspective::switch_to_source_code),
+ ActionEntry::DEFAULT,
+ "",
+ false
+ },
};
static ui_utils::ActionEntry s_debugger_busy_action_entries [] = {
@@ -3779,6 +3870,39 @@ DBGPerspective::clear_session_data ()
m_priv->source_dirs.clear ();
}
+bool
+DBGPerspective::find_absolute_path (const UString& a_file_path,
+ UString &a_absolute_path)
+{
+ // First, assume it's a full path name already.
+ if (Glib::file_test (a_file_path, Glib::FILE_TEST_IS_REGULAR)) {
+ a_absolute_path = a_file_path;
+ return true;
+ }
+ // If that didn't work, look for a file of that name in the search
+ // directories.
+ if (find_file_in_source_dirs (a_file_path, a_absolute_path)) {
+ return true;
+ }
+ return false;
+}
+
+bool
+DBGPerspective::find_absolute_path_or_ask_user (const UString& a_file_path,
+ UString& a_absolute_path)
+{
+ if (!find_absolute_path (a_file_path, a_absolute_path)) {
+ if (ask_user_to_select_file (a_file_path, a_absolute_path)) {
+ UString parent_dir = Glib::filename_to_utf8
+ (Glib::path_get_dirname (a_absolute_path));
+ m_priv->search_paths.push_back (parent_dir);
+ } else {
+ return false;
+ }
+ }
+ return true;
+}
+
void
DBGPerspective::append_source_editor (SourceEditor &a_sv,
const UString &a_path)
@@ -3874,7 +3998,10 @@ DBGPerspective::append_source_editor (SourceEditor &a_sv,
&DBGPerspective::on_leave_notify_event_signal));
}
- m_priv->opened_file_action_group->set_sensitive (true);
+ if (get_num_notebook_pages () == 1) {
+ m_priv->opened_file_action_group->set_sensitive (true);
+ update_src_dependant_bp_actions_sensitiveness ();
+ }
}
SourceEditor*
@@ -4892,6 +5019,9 @@ DBGPerspective::get_file_mime_type (const UString &a_path,
NEMIVER_CATCH_AND_RETURN (false)
}
+// Make sure the source buffer a_buf is properly setup to have the mime
+// type a_mime_type. If a_buf is null, a new one is created.
+// Returns true upon successful completion, false otherwise.
bool
DBGPerspective::setup_buffer_mime_and_lang (Glib::RefPtr<SourceBuffer> &a_buf,
const std::string &a_mime_type)
@@ -5050,7 +5180,7 @@ DBGPerspective::create_source_editor (Glib::RefPtr<SourceBuffer> &a_source_buf,
a_source_buf,
true));
if (!a_current_address.empty ()) {
- source_editor->composite_buf_loc_to_line (a_current_address.raw (),
+ source_editor->assembly_buf_loc_to_line (a_current_address.raw (),
current_line);
}
} else {
@@ -5141,7 +5271,8 @@ DBGPerspective::open_file (const UString &a_path,
int a_current_line)
{
RETURN_VAL_IF_FAIL (m_priv, false);
- RETURN_VAL_IF_FAIL (!a_path.empty (), false);
+ if (a_path.empty ())
+ return false;
if (get_source_editor_from_path (a_path)) {return true;}
@@ -5279,6 +5410,11 @@ DBGPerspective::load_disassembly (const IDebugger::DisassembleInfo &/*a_info*/,
NEMIVER_CATCH_AND_RETURN (false)
}
+// If new disassembly dedicated tab was already present in the perspective,
+// create a new one, otherwise, reuse the one that was already present.
+// Then load the assembly insns a_asm described by a_info into the
+// source buffer of the disassembly tab.
+// Return true upon successful completion, false otherwise.
bool
DBGPerspective::open_disassembly (const IDebugger::DisassembleInfo &a_info,
const std::list<IDebugger::AsmInstr> &a_asm)
@@ -5288,25 +5424,108 @@ DBGPerspective::open_disassembly (const IDebugger::DisassembleInfo &a_info,
NEMIVER_TRY
Glib::RefPtr<SourceBuffer> source_buffer;
+
+ SourceEditor *source_editor =
+ get_source_editor_from_path (get_disassembly_title ());
+
+ if (source_editor) {
+ source_buffer = source_editor->source_view ().get_source_buffer ();
+ source_buffer->erase (source_buffer->begin (), source_buffer->end ());
+ }
if (!load_disassembly (a_info, a_asm, source_buffer)) {
return false;
}
-
- SourceEditor *source_editor =
- create_source_editor (source_buffer,
- /*a_disassembly_view=*/true,
- get_disassembly_title (),
- -1,
- /*a_current_address=*/"");
- THROW_IF_FAIL (source_editor);
- source_editor->show_all ();
- append_source_editor (*source_editor, get_disassembly_title ());
+ if (!source_editor) {
+ source_editor =
+ create_source_editor (source_buffer,
+ /*a_disassembly_view=*/true,
+ get_disassembly_title (),
+ -1,
+ /*a_current_address=*/"");
+ THROW_IF_FAIL (source_editor);
+ source_editor->show_all ();
+ append_source_editor (*source_editor, get_disassembly_title ());
+ }
NEMIVER_CATCH_AND_RETURN (false);
return true;
}
+// Get the source editor of the source file being currently debugged,
+// switch it into the disassembly mode and load the asm insns
+// represented by a_info and a_asm into its source buffer.
+// \param a_info descriptor of the assembly instructions
+// \param a_asm a list of asm instructions.
+void
+DBGPerspective::switch_to_disassembly
+ (const IDebugger::DisassembleInfo &a_info,
+ const std::list<IDebugger::AsmInstr> &a_asm)
+{
+ LOG_FUNCTION_SCOPE_NORMAL_DD;
+
+ SourceEditor *source_editor = get_current_source_editor ();
+ if (source_editor == 0)
+ return;
+
+ Glib::RefPtr<SourceBuffer> asm_buf;
+ if ((asm_buf = source_editor->get_assembly_source_buffer ()) == 0) {
+ setup_buffer_mime_and_lang (asm_buf, "test/x-asm");
+ source_editor->register_assembly_source_buffer (asm_buf);
+ asm_buf = source_editor->get_assembly_source_buffer ();
+ RETURN_IF_FAIL (asm_buf);
+ }
+ if (!load_disassembly (a_info, a_asm, asm_buf)) {
+ LOG_ERROR ("failed to load asm");
+ return;
+ }
+ if (!source_editor->switch_to_assembly_source_buffer ()) {
+ LOG_ERROR ("Could not switch the current view to asm");
+ }
+}
+
+// Get the source editor of the source file being currently debugged,
+// switch it into the source code mode. If necessary, (re) load the
+// source code.
+void
+DBGPerspective::switch_to_source_code ()
+{
+ SourceEditor *source_editor = get_current_source_editor ();
+ if (source_editor == 0)
+ return;
+ Glib::RefPtr<SourceBuffer> source_buf;
+ UString source_path;
+ if ((source_buf = source_editor->get_non_assembly_source_buffer ()) == 0) {
+ // Woops!
+ // We don't have any source code buffer. Let's try hard to get
+ // the source code corresponding to the current frame. For that,
+ // we'll hope to have proper debug info for the binary being
+ // debugged, and the source code available on disk.
+ if (m_priv->current_frame.address ().empty ()) {
+ LOG_DD ("No current instruction pointer");
+ return;
+ }
+ if (m_priv->current_frame.file_name ().empty ()) {
+ LOG_DD ("No file name information for current frame");
+ return;
+ }
+ UString absolute_path, mime_type;
+ if (!find_absolute_path_or_ask_user (m_priv->current_frame.file_name (),
+ absolute_path)) {
+ LOG_DD ("Could not find file: "
+ << m_priv->current_frame.file_name ());
+ return;
+ }
+ get_file_mime_type (absolute_path, mime_type);
+ setup_buffer_mime_and_lang (source_buf, mime_type);
+ load_file (absolute_path, source_buf);
+ source_editor->register_non_assembly_source_buffer (source_buf);
+ }
+ source_editor->switch_to_non_assembly_source_buffer ();
+ source_editor->get_path (source_path);
+ apply_decorations_to_text (source_path);
+}
+
Gtk::Widget*
DBGPerspective::load_menu (const UString &a_filename,
const UString &a_widget_name)
@@ -6155,10 +6374,42 @@ DBGPerspective::append_visual_breakpoint (const UString &a_file_name,
actual_file_path, enabled);
}
+// Popup a dialog asking the user to select the file a_file_name.
+// \param a_file_name the name of the file we want the user to select.
+// \param a_selected_file_path the path to the file the user actually
+// selected.
+// \return true if the user really selected the file we wanted, false
+// otherwise.
+bool
+DBGPerspective::ask_user_to_select_file (const UString &a_file_name,
+ UString &a_selected_file_path)
+{
+ LocateFileDialog dialog (plugin_path (), a_file_name);
+ // start looking in the working directory
+ dialog.file_location (m_priv->prog_cwd);
+ int result = dialog.run ();
+ if (result == Gtk::RESPONSE_OK) {
+ UString file_path = dialog.file_location ();
+ if (!Glib::file_test (file_path, Glib::FILE_TEST_IS_REGULAR)
+ || (Glib::path_get_basename (a_file_name)
+ != Glib::path_get_basename (file_path)))
+ return false;
+ UString parent_dir =
+ Glib::filename_to_utf8 (Glib::path_get_dirname
+ (dialog.file_location ()));
+ if (!Glib::file_test (parent_dir, Glib::FILE_TEST_IS_DIR))
+ return false;
+
+ a_selected_file_path = file_path;
+ return true;
+ }
+ return false;
+}
+
bool
DBGPerspective::append_visual_breakpoint (const UString &a_file_name,
int a_linenum,
- UString &a_actual_file_name,
+ UString &a_actual_file_path,
bool enabled)
{
if (a_file_name.empty()) {
@@ -6173,105 +6424,32 @@ DBGPerspective::append_visual_breakpoint (const UString &a_file_name,
if (a_linenum < 0) {a_linenum = 0;}
- UString actual_file_name;
+ UString actual_file_path;
SourceEditor *source_editor =
get_source_editor_from_path (a_file_name,
- actual_file_name);
- //first assume that it's a full pathname and just try to open it
- if (!source_editor) {
- if (Glib::file_test (a_file_name, Glib::FILE_TEST_IS_REGULAR)) {
- if (!open_file (a_file_name)) {
- UString msg;
- msg.printf (_("Failed to open file %s. "
- "Would you like to open another file which "
- "would have the same content ?"),
- a_file_name.c_str ());
- if (ui_utils::ask_yes_no_question (msg)
- == Gtk::RESPONSE_YES) {
- LocateFileDialog dialog (plugin_path (), a_file_name);
- dialog.file_location (m_priv->prog_cwd);
- int result = dialog.run ();
- if (result == Gtk::RESPONSE_OK) {
- UString file_path = dialog.file_location ();
- THROW_IF_FAIL (Glib::file_test (file_path,
- Glib::FILE_TEST_IS_REGULAR));
- std::string raw_dir = Glib::path_get_dirname
- (dialog.file_location ());
- UString parent_dir = Glib::filename_to_utf8 (raw_dir);
- THROW_IF_FAIL (Glib::file_test (raw_dir,
- Glib::FILE_TEST_IS_DIR));
- m_priv->search_paths.push_back (parent_dir);
- if (!open_file (file_path)) {
- return false;
- }
- source_editor =
- get_source_editor_from_path (file_path,
- actual_file_name);
- } else {
- return false;
- }
- } else {
- return false;
- }
+ actual_file_path);
+ if (source_editor == 0) {
+ if (!find_absolute_path (a_file_name, actual_file_path)) {
+ if (ask_user_to_select_file (a_file_name, actual_file_path)) {
+ UString parent_dir =
+ Glib::filename_to_utf8
+ (Glib::path_get_dirname (actual_file_path));
+ m_priv->search_paths.push_back (parent_dir);
}
- source_editor = get_source_editor_from_path (a_file_name,
- actual_file_name);
}
- }
- //if that didn't work, look for an opened source editor
- //that matches the base name
- if (!source_editor) {
+ open_file (actual_file_path);
source_editor = get_source_editor_from_path (a_file_name,
- actual_file_name,
- true);
+ actual_file_path);
}
- //if that still didn't work, look for a file of that name in the search
- //directories and then as a last resort ask the user to locate it
- //manually
- if (!source_editor) {
- UString file_path;
- if (!find_file_in_source_dirs (a_file_name, file_path)) {
- LOG_DD ("didn't find file either in opened files "
- " or in source dirs:");
- //Pop up a dialog asking user to select the specified file
- LocateFileDialog dialog (plugin_path (), a_file_name);
- //start looking in the working directory
- dialog.file_location (m_priv->prog_cwd);
- int result = dialog.run ();
- if (result == Gtk::RESPONSE_OK) {
- file_path = dialog.file_location ();
- THROW_IF_FAIL (Glib::file_test (file_path,
- Glib::FILE_TEST_IS_REGULAR));
- THROW_IF_FAIL (Glib::path_get_basename(a_file_name) ==
- Glib::path_get_basename(file_path));
- UString parent_dir = Glib::filename_to_utf8
- (Glib::path_get_dirname (dialog.file_location ()));
- THROW_IF_FAIL (Glib::file_test
- (parent_dir, Glib::FILE_TEST_IS_DIR));
-
- //Also add the parent directory to the list
- //of paths to search for this session so
- //you don't have to keep selecting files if
- //they're all in the same directory.
- //We can assume that the parent
- //directory is not already in the
- //list of source dirs
- //(or else it would have been found and the
- //user wouldn't have had to locate it manually)
- //so just stack it
- //on the end of the list
- m_priv->search_paths.push_back (parent_dir);
- }
- }
-
- if (!file_path.empty ()) {
- open_file (file_path);
- source_editor = get_source_editor_from_path (file_path,
- actual_file_name);
- }
+ // if that didn't work, look for an opened source editor
+ // that matches the base name
+ if (source_editor == 0) {
+ source_editor = get_source_editor_from_path (a_file_name,
+ actual_file_path,
+ /*basename_only=*/true);
}
- //finally, if none of these things worked, display an error
+ // finally, if none of these things worked, display an error
if (!source_editor) {
LOG_ERROR ("Could not find source editor for file: '"
<< a_file_name
@@ -6282,7 +6460,7 @@ DBGPerspective::append_visual_breakpoint (const UString &a_file_name,
source_editor->set_visual_breakpoint_at_line (a_linenum, enabled);
}
- a_actual_file_name = actual_file_name;
+ a_actual_file_path = actual_file_path;
return true;
}
@@ -6349,6 +6527,16 @@ DBGPerspective::choose_function_overload
debugger ()->choose_function_overloads (nums);
}
+void
+DBGPerspective::remove_visual_decorations_from_text
+ (const UString &a_file_path)
+{
+ SourceEditor *editor = get_source_editor_from_path (a_file_path);
+ if (editor == 0)
+ return;
+ editor->clear_decorations ();
+}
+
bool
DBGPerspective::apply_decorations_to_text (const UString &a_file_path)
{
@@ -6616,6 +6804,42 @@ DBGPerspective::refresh_locals ()
}
void
+DBGPerspective::disassemble (bool a_show_asm_in_new_tab)
+{
+ THROW_IF_FAIL (m_priv);
+
+ // If we don't have the current instruction pointer (IP), there is
+ // nothing we can do.
+ if (!debugger ()->is_attached_to_target ()
+ || m_priv->current_frame.address ().empty ()) {
+ LOG_DD ("No current instruction pointer");
+ return;
+ }
+
+ sigc::slot<void,
+ const IDebugger::DisassembleInfo,
+ const std::list<IDebugger::AsmInstr>& > s;
+ if (a_show_asm_in_new_tab)
+ s = sigc::bind (sigc::mem_fun (this,
+ &DBGPerspective::on_debugger_disassembled_signal),
+ true);
+
+ else
+ s = sigc::bind
+ (sigc::mem_fun
+ (this,
+ &DBGPerspective::on_debugger_disassembled_signal),
+ false);
+
+ debugger ()->disassemble
+ (0 /* Start @ to disassemble */,
+ true /* Start @ is relative to current IP */,
+ NUM_INSTR_TO_DISASSEMBLE * sizeof (void *),
+ true /* End @ is relative to current IP */,
+ s);
+}
+
+void
DBGPerspective::toggle_breakpoint_enabled ()
{
LOG_FUNCTION_SCOPE_NORMAL_DD;
diff --git a/src/uicommon/nmv-source-editor.cc b/src/uicommon/nmv-source-editor.cc
index 624cde8..624129c 100644
--- a/src/uicommon/nmv-source-editor.cc
+++ b/src/uicommon/nmv-source-editor.cc
@@ -173,7 +173,7 @@ struct SourceEditor::Priv {
Gtk::Label *line_col_label;
Gtk::HBox *status_box;
- struct NonCompositeBufContext {
+ struct NonAssemblyBufContext {
Glib::RefPtr<SourceBuffer> buffer;
int current_column;
int current_line;
@@ -181,7 +181,7 @@ struct SourceEditor::Priv {
sigc::signal<void, int, bool> marker_region_got_clicked_signal;
UString path;
- NonCompositeBufContext (Glib::RefPtr<SourceBuffer> a_buf,
+ NonAssemblyBufContext (Glib::RefPtr<SourceBuffer> a_buf,
int a_cur_col, int a_cur_line) :
buffer (a_buf),
current_column (a_cur_col),
@@ -189,72 +189,72 @@ struct SourceEditor::Priv {
{
}
- NonCompositeBufContext (int a_cur_col, int a_cur_line) :
+ NonAssemblyBufContext (int a_cur_col, int a_cur_line) :
current_column (a_cur_col),
current_line (a_cur_line)
{
}
- NonCompositeBufContext () :
+ NonAssemblyBufContext () :
current_column (-1),
current_line (-1)
{
}
- } non_comp_ctxt;
+ } non_asm_ctxt;
- struct CompositeBufContext {
+ struct AssemblyBufContext {
Glib::RefPtr<SourceBuffer> buffer;
Line2AddrFunc line_to_locus_func;
Addr2LineFunc locus_to_line_func;
- CompositeBufContext ()
+ AssemblyBufContext ()
{
}
- CompositeBufContext
+ AssemblyBufContext
(Glib::RefPtr<SourceBuffer> a_buf) :
buffer (a_buf)
{
}
- } comp_ctxt;
+ } asm_ctxt;
sigc::signal<void, const Gtk::TextBuffer::iterator&>
insertion_changed_signal;
- void register_composite_source_buffer
+ void register_assembly_source_buffer
(Glib::RefPtr<SourceBuffer> &a_buf)
{
- comp_ctxt.buffer = a_buf;
+ asm_ctxt.buffer = a_buf;
}
- void register_non_composite_source_buffer
+ void register_non_assembly_source_buffer
(Glib::RefPtr<SourceBuffer> &a_buf)
{
- non_comp_ctxt.buffer = a_buf;
+ non_asm_ctxt.buffer = a_buf;
}
- bool switch_to_composite_source_buffer ()
+ bool switch_to_assembly_source_buffer ()
{
RETURN_VAL_IF_FAIL (source_view, false);
- if (comp_ctxt.buffer
+ if (asm_ctxt.buffer
&& (source_view->get_source_buffer ()
- != comp_ctxt.buffer)) {
- source_view->set_source_buffer (comp_ctxt.buffer);
+ != asm_ctxt.buffer)) {
+ source_view->set_source_buffer (asm_ctxt.buffer);
return true;
}
return false;
}
- bool switch_to_non_composite_source_buffer ()
+ bool switch_to_non_assembly_source_buffer ()
{
RETURN_VAL_IF_FAIL (source_view, false);
- if (comp_ctxt.buffer
+ if (asm_ctxt.buffer
&& (source_view->get_source_buffer ()
- != non_comp_ctxt.buffer)) {
- source_view->set_source_buffer (non_comp_ctxt.buffer);
+ != non_asm_ctxt.buffer)) {
+ source_view->set_source_buffer (non_asm_ctxt.buffer);
return true;
}
return false;
@@ -265,7 +265,7 @@ struct SourceEditor::Priv {
//**************
void on_marker_region_got_clicked (int a_line, bool a_dialog_requested)
{
- non_comp_ctxt.marker_region_got_clicked_signal.emit
+ non_asm_ctxt.marker_region_got_clicked_signal.emit
(a_line, a_dialog_requested);
}
@@ -288,8 +288,8 @@ struct SourceEditor::Priv {
void on_signal_insertion_moved (gint a_line, gint a_col)
{
if (a_line || a_col) {}
- non_comp_ctxt.current_line = a_line;
- non_comp_ctxt.current_column = a_col;
+ non_asm_ctxt.current_line = a_line;
+ non_asm_ctxt.current_column = a_col;
update_line_col_label ();
}
@@ -321,7 +321,7 @@ struct SourceEditor::Priv {
(sigc::mem_fun (*this, &SourceEditor::Priv::on_mark_set_signal));
source_view->get_buffer ()->signal_insert ().connect
(sigc::mem_fun (*this, &SourceEditor::Priv::on_signal_insert));
- non_comp_ctxt.signal_insertion_moved.connect
+ non_asm_ctxt.signal_insertion_moved.connect
(sigc::mem_fun (*this,
&SourceEditor::Priv::on_signal_insertion_moved));
source_view->get_buffer ()->signal_mark_set ().connect
@@ -330,11 +330,11 @@ struct SourceEditor::Priv {
void update_line_col_info_from_iter (const Gtk::TextBuffer::iterator &a_iter)
{
- non_comp_ctxt.current_line = a_iter.get_line () + 1;
- non_comp_ctxt.current_column = get_column_from_iter (a_iter);
- non_comp_ctxt.signal_insertion_moved.emit
- (non_comp_ctxt.current_line,
- non_comp_ctxt.current_column);
+ non_asm_ctxt.current_line = a_iter.get_line () + 1;
+ non_asm_ctxt.current_column = get_column_from_iter (a_iter);
+ non_asm_ctxt.signal_insertion_moved.emit
+ (non_asm_ctxt.current_line,
+ non_asm_ctxt.current_column);
}
void update_line_col_label ()
@@ -345,8 +345,8 @@ struct SourceEditor::Priv {
}
UString message;
message.printf (_("Line: %i, Column: %i"),
- non_comp_ctxt.current_line,
- non_comp_ctxt.current_column);
+ non_asm_ctxt.current_line,
+ non_asm_ctxt.current_column);
line_col_label->set_text (message);
}
@@ -413,7 +413,7 @@ struct SourceEditor::Priv {
source_view (Gtk::manage (new SourceView)),
line_col_label (Gtk::manage (new Gtk::Label)),
status_box (Gtk::manage (new Gtk::HBox)),
- non_comp_ctxt (-1, -1)
+ non_asm_ctxt (-1, -1)
{
init ();
@@ -421,17 +421,17 @@ struct SourceEditor::Priv {
explicit Priv (const UString &a_root_dir,
Glib::RefPtr<SourceBuffer> &a_buf,
- bool a_composite) :
+ bool a_assembly) :
root_dir (a_root_dir),
source_view (Gtk::manage (new SourceView (a_buf))),
line_col_label (Gtk::manage (new Gtk::Label)),
status_box (Gtk::manage (new Gtk::HBox)),
- non_comp_ctxt (a_buf, -1, -1)
+ non_asm_ctxt (a_buf, -1, -1)
{
- if (a_composite) {
- comp_ctxt.buffer = a_buf;
+ if (a_assembly) {
+ asm_ctxt.buffer = a_buf;
} else {
- non_comp_ctxt.buffer = a_buf;
+ non_asm_ctxt.buffer = a_buf;
}
init ();
}
@@ -441,7 +441,7 @@ struct SourceEditor::Priv {
root_dir (a_root_dir),
source_view (Gtk::manage (new SourceView (a_buf))),
status_box (Gtk::manage (new Gtk::HBox)),
- comp_ctxt (a_buf)
+ asm_ctxt (a_buf)
{
init ();
}
@@ -485,9 +485,9 @@ SourceEditor::SourceEditor ()
SourceEditor::SourceEditor (const UString &a_root_dir,
Glib::RefPtr<SourceBuffer> &a_buf,
- bool a_composite)
+ bool a_assembly)
{
- m_priv.reset (new Priv (a_root_dir, a_buf, a_composite));
+ m_priv.reset (new Priv (a_root_dir, a_buf, a_assembly));
init ();
}
@@ -506,26 +506,26 @@ SourceEditor::source_view () const
gint
SourceEditor::current_line () const
{
- return m_priv->non_comp_ctxt.current_line;
+ return m_priv->non_asm_ctxt.current_line;
}
void
SourceEditor::current_line (gint &a_line)
{
- m_priv->non_comp_ctxt.current_line = a_line;
+ m_priv->non_asm_ctxt.current_line = a_line;
}
gint
SourceEditor::current_column () const
{
- return m_priv->non_comp_ctxt.current_column;
+ return m_priv->non_asm_ctxt.current_column;
}
void
SourceEditor::current_column (gint &a_col)
{
LOG_DD ("current colnum " << (int) a_col);
- m_priv->non_comp_ctxt.current_column = a_col;
+ m_priv->non_asm_ctxt.current_column = a_col;
}
void
@@ -665,6 +665,30 @@ SourceEditor::remove_visual_breakpoint_from_line (int a_line)
m_priv->markers.erase (iter);
}
+void
+SourceEditor::clear_decorations ()
+{
+#ifdef WITH_SOURCEVIEWMM2
+ std::map<int, Glib::RefPtr<gtksourceview::SourceMark> >::iterator it;
+#else
+ std::map<int, Glib::RefPtr<gtksourceview::SourceMarker> >::iterator it;
+#endif // WITH_SOURCEVIEWMM2
+
+ // Clear breakpoint markers
+ for (it = m_priv->markers.begin (); it != m_priv->markers.end (); ++it) {
+ if (!it->second->get_deleted ()) {
+#ifdef WITH_SOURCEVIEWMM2
+ source_view ().get_source_buffer ()->delete_mark (it->second);
+#else
+ source_view ().get_source_buffer ()->delete_marker (it->second);
+#endif // WITH_SOURCEVIEWMM2
+ m_priv->markers.erase (it);
+ }
+ }
+
+ unset_where_marker ();
+}
+
bool
SourceEditor::is_visual_breakpoint_set_at_line (int a_line) const
{
@@ -732,20 +756,20 @@ SourceEditor::scroll_to_iter (Gtk::TextIter &a_iter)
void
SourceEditor::set_path (const UString &a_path)
{
- m_priv->non_comp_ctxt.path = a_path;
+ m_priv->non_asm_ctxt.path = a_path;
}
void
SourceEditor::get_path (UString &a_path) const
{
- a_path = m_priv->non_comp_ctxt.path;
+ a_path = m_priv->non_asm_ctxt.path;
}
void
SourceEditor::get_file_name (UString &a_file_name)
{
string path;
- path = Glib::locale_from_utf8 (m_priv->non_comp_ctxt.path);
+ path = Glib::locale_from_utf8 (m_priv->non_asm_ctxt.path);
path = Glib::path_get_basename (path);
a_file_name = Glib::locale_to_utf8 (path);
}
@@ -901,96 +925,96 @@ SourceEditor::do_search (const UString &a_str,
return false;
}
-/// Registers a composite source buffer
-/// \param a_buf the composite source buffer
+/// Registers a assembly source buffer
+/// \param a_buf the assembly source buffer
/// \param a_line_to_locus_func a unary function that converts a line
/// number into a meaningful location for this source buffer.
/// \param a_locus_to_line_func a unary function that converst a
/// meaningful location into a line buffer.
void
-SourceEditor::register_composite_source_buffer
+SourceEditor::register_assembly_source_buffer
(Glib::RefPtr<SourceBuffer> &a_buf)
{
- m_priv->register_composite_source_buffer (a_buf);
+ m_priv->register_assembly_source_buffer (a_buf);
}
-/// Registers a normal (non-composite) source buffer.
+/// Registers a normal (non-assembly) source buffer.
/// \param a_buf the source buffer to register.
void
-SourceEditor::register_non_composite_source_buffer
+SourceEditor::register_non_assembly_source_buffer
(Glib::RefPtr<SourceBuffer> &a_buf)
{
- m_priv->register_non_composite_source_buffer (a_buf);
+ m_priv->register_non_assembly_source_buffer (a_buf);
}
-/// Get the composite source buffer that was registered, or a NULL
+/// Get the assembly source buffer that was registered, or a NULL
/// pointer if no one was registered before.
/// \return a smart pointer to the source buffer.
Glib::RefPtr<SourceBuffer>
-SourceEditor::get_composite_source_buffer () const
+SourceEditor::get_assembly_source_buffer () const
{
- return m_priv->comp_ctxt.buffer;
+ return m_priv->asm_ctxt.buffer;
}
-/// Get the non-composite source buffer that was registered, or a NULL
+/// Get the non-assembly source buffer that was registered, or a NULL
/// pointer if no one was registered before.
/// \return a smart pointer to the source buffer.
Glib::RefPtr<SourceBuffer>
-SourceEditor::get_non_composite_source_buffer () const
+SourceEditor::get_non_assembly_source_buffer () const
{
- return m_priv->non_comp_ctxt.buffer;
+ return m_priv->non_asm_ctxt.buffer;
}
-/// Switch the editor to the composite source buffer that was
-/// registered. This function has no effect if no composite buffer was
+/// Switch the editor to the assembly source buffer that was
+/// registered. This function has no effect if no assembly buffer was
/// registered.
/// \return true if the switch was done, false otherwise.
bool
-SourceEditor::switch_to_composite_source_buffer ()
+SourceEditor::switch_to_assembly_source_buffer ()
{
- return m_priv->switch_to_composite_source_buffer ();
+ return m_priv->switch_to_assembly_source_buffer ();
}
-/// Switch the editor to the non-composite source buffer that was
-/// registered. This function has no effect if no non-composite source
+/// Switch the editor to the non-assembly source buffer that was
+/// registered. This function has no effect if no non-assembly source
/// buffer was registered.
/// \return true if the switch was done, false otherwise.
bool
-SourceEditor::switch_to_non_composite_source_buffer ()
+SourceEditor::switch_to_non_assembly_source_buffer ()
{
RETURN_VAL_IF_FAIL (m_priv && m_priv->source_view, false);
- if (m_priv->comp_ctxt.buffer
+ if (m_priv->asm_ctxt.buffer
&& (m_priv->source_view->get_source_buffer ()
- != m_priv->non_comp_ctxt.buffer)) {
- m_priv->source_view->set_source_buffer (m_priv->non_comp_ctxt.buffer);
+ != m_priv->non_asm_ctxt.buffer)) {
+ m_priv->source_view->set_source_buffer (m_priv->non_asm_ctxt.buffer);
return true;
}
return false;
}
bool
-SourceEditor::composite_buf_loc_to_line (const Address &a_addr, int &a_line)
+SourceEditor::assembly_buf_loc_to_line (const Address &a_addr, int &a_line)
{
- Glib::RefPtr<SourceBuffer> buf = get_composite_source_buffer ();
+ Glib::RefPtr<SourceBuffer> buf = get_assembly_source_buffer ();
RETURN_VAL_IF_FAIL (buf, false);
- a_line = m_priv->comp_ctxt.locus_to_line_func (buf, a_addr);
+ a_line = m_priv->asm_ctxt.locus_to_line_func (buf, a_addr);
return true;
}
bool
-SourceEditor::composite_buf_line_to_loc (int a_line, Address &a_address)
+SourceEditor::assembly_buf_line_to_loc (int a_line, Address &a_address)
{
- Glib::RefPtr<SourceBuffer> buf = get_composite_source_buffer ();
+ Glib::RefPtr<SourceBuffer> buf = get_assembly_source_buffer ();
RETURN_VAL_IF_FAIL (buf, false);
- a_address = m_priv->comp_ctxt.line_to_locus_func (buf, a_line);
+ a_address = m_priv->asm_ctxt.line_to_locus_func (buf, a_line);
return true;
}
sigc::signal<void, int, bool>&
SourceEditor::marker_region_got_clicked_signal () const
{
- return m_priv->non_comp_ctxt.marker_region_got_clicked_signal;
+ return m_priv->non_asm_ctxt.marker_region_got_clicked_signal;
}
sigc::signal<void, const Gtk::TextBuffer::iterator&>&
diff --git a/src/uicommon/nmv-source-editor.h b/src/uicommon/nmv-source-editor.h
index 4d660db..a9c59bf 100644
--- a/src/uicommon/nmv-source-editor.h
+++ b/src/uicommon/nmv-source-editor.h
@@ -75,6 +75,7 @@ public:
void unset_where_marker ();
void set_visual_breakpoint_at_line (int a_line, bool enabled=true);
void remove_visual_breakpoint_from_line (int a_line);
+ void clear_decorations ();
bool is_visual_breakpoint_set_at_line (int a_line) const;
void scroll_to_line (int a_line);
void scroll_to_iter (Gtk::TextIter &a_iter);
@@ -98,35 +99,35 @@ public:
/// \name Composite Source buffer handling.
/// @{
- /// A composite buffer is a buffer which content doesn't come
+ /// A assembly buffer is a buffer which content doesn't come
/// directly from a file. It as been composed in memory. We use
- /// composite buffers to represent the assembly view of a text file
+ /// assemyb buffers to represent the assembly view of a text file
/// being debugged.
- /// Unlike non-composite buffers, meaningful locations inside the buffer
- /// are not necessarily line numbers. They can be something else.
- /// In the case of assembly view, a meaningful location is the address
+ /// Unlike non-assembly buffers, meaningful locations inside the buffer
+ /// are not necessarily line numbers.
+ /// In an assembly view, a meaningful location is the address
/// of a machine instruction. So there somehow must be a kind of mapping
- /// between the location used for the composite buffer and the actual
+ /// between the location used for the assembly buffer and the actual
/// line number, because the underlying SourceBuffer implementation
/// relies on line numbers anyhow.
- void register_composite_source_buffer
+ void register_assembly_source_buffer
(Glib::RefPtr<SourceBuffer> &a_buf);
- void register_non_composite_source_buffer
+ void register_non_assembly_source_buffer
(Glib::RefPtr<SourceBuffer> &a_buf);
- Glib::RefPtr<SourceBuffer> get_composite_source_buffer () const;
+ Glib::RefPtr<SourceBuffer> get_assembly_source_buffer () const;
- Glib::RefPtr<SourceBuffer> get_non_composite_source_buffer () const;
+ Glib::RefPtr<SourceBuffer> get_non_assembly_source_buffer () const;
- bool switch_to_composite_source_buffer ();
+ bool switch_to_assembly_source_buffer ();
- bool switch_to_non_composite_source_buffer ();
+ bool switch_to_non_assembly_source_buffer ();
- bool composite_buf_loc_to_line (const Address &, int &);
+ bool assembly_buf_loc_to_line (const Address &, int &);
- bool composite_buf_line_to_loc (int, Address &);
+ bool assembly_buf_line_to_loc (int, Address &);
/// @}
/// \name signals
[
Date Prev][
Date Next] [
Thread Prev][
Thread Next]
[
Thread Index]
[
Date Index]
[
Author Index]