diff --git a/angrmanagement/ui/icons.py b/angrmanagement/ui/icons.py index b12a8655b5..afd2c33d1d 100644 --- a/angrmanagement/ui/icons.py +++ b/angrmanagement/ui/icons.py @@ -17,6 +17,7 @@ "docs": "mdi6.book-open-page-variant", "file": "mdi.file", "file-open": "mdi.folder-open", + "file-save": "mdi.floppy", "functions-view": "mdi.function", "hex-view": "mdi.hexadecimal", "jobs-view": "fa5s.hammer", @@ -30,6 +31,7 @@ "strings-view": "msc.symbol-string", "traces-view": "mdi.go-kart-track", "types-view": "msc.symbol-class", + "explorer-view": "msc.globe", } diff --git a/angrmanagement/ui/main_window.py b/angrmanagement/ui/main_window.py index 969c6b6e82..dee523fbb1 100644 --- a/angrmanagement/ui/main_window.py +++ b/angrmanagement/ui/main_window.py @@ -501,6 +501,7 @@ def _register_commands(self) -> None: ("View: Disassembly (Graph)", self.workspace.show_graph_disassembly_view), ("View: Disassembly (Linear)", self.workspace.show_linear_disassembly_view), ("View: Functions", self.workspace.show_functions_view), + ("View: Explorer", self.workspace.show_explorer_view), ("View: Hex", self.workspace.show_hex_view), ("View: Interaction", self.workspace.show_interaction_view), ("View: Log", self.workspace.show_log_view), diff --git a/angrmanagement/ui/menus/file_menu.py b/angrmanagement/ui/menus/file_menu.py index ada4b8cc19..06ba2bf998 100644 --- a/angrmanagement/ui/menus/file_menu.py +++ b/angrmanagement/ui/menus/file_menu.py @@ -26,7 +26,7 @@ class RecentMenuEntry(MenuEntry): def __init__(self, path) -> None: self.path = path - super().__init__(path, self.action_target) + super().__init__(path, self.action_target, icon=icon("file")) def action_target(self) -> None: GlobalInfo.main_window.load_file(self.path) @@ -42,8 +42,18 @@ def __init__(self, main_window: MainWindow) -> None: self._project = main_window.workspace.main_instance.project self._save_entries = [ - MenuEntry("&Save angr database...", main_window.save_database, shortcut=QKeySequence("Ctrl+S")), - MenuEntry("S&ave angr database as...", main_window.save_database_as, shortcut=QKeySequence("Ctrl+Shift+S")), + MenuEntry( + "&Save angr database...", + main_window.save_database, + shortcut=QKeySequence("Ctrl+S"), + icon=icon("file-save"), + ), + MenuEntry( + "S&ave angr database as...", + main_window.save_database_as, + shortcut=QKeySequence("Ctrl+Shift+S"), + icon=icon("file-save"), + ), MenuEntry("Save patched binary as...", main_window.save_patched_binary_as), ] self._edit_save() @@ -52,7 +62,12 @@ def __init__(self, main_window: MainWindow) -> None: self.recent_menu = Menu("Load recent") self.entries.extend( [ - MenuEntry("L&oad a new binary...", main_window.open_file_button, shortcut=QKeySequence("Ctrl+O")), + MenuEntry( + "L&oad a new binary...", + main_window.open_file_button, + shortcut=QKeySequence("Ctrl+O"), + icon=icon("file-open"), + ), *( [] if archr is None @@ -71,7 +86,12 @@ def __init__(self, main_window: MainWindow) -> None: ), self.recent_menu, MenuSeparator(), - MenuEntry("&Load angr database...", main_window.load_database, shortcut=QKeySequence("Ctrl+L")), + MenuEntry( + "&Load angr database...", + main_window.load_database, + shortcut=QKeySequence("Ctrl+L"), + icon=icon("file-open"), + ), *self._save_entries, MenuSeparator(), MenuEntry("Load a new &trace...", main_window.load_trace), diff --git a/angrmanagement/ui/menus/view_menu.py b/angrmanagement/ui/menus/view_menu.py index 3b5b19e96f..4e125e0241 100644 --- a/angrmanagement/ui/menus/view_menu.py +++ b/angrmanagement/ui/menus/view_menu.py @@ -128,6 +128,7 @@ def __init__(self, main_window: MainWindow) -> None: MenuEntry("&Patches", main_window.workspace.show_patches_view, icon=icon("patches-view")), MenuEntry("&Types", main_window.workspace.show_types_view, icon=icon("types-view")), MenuEntry("&Functions", main_window.workspace.show_functions_view, icon=icon("functions-view")), + MenuEntry("&Explorer", main_window.workspace.show_explorer_view, icon=icon("explorer-view")), MenuEntry("&Traces", main_window.workspace.show_traces_view, icon=icon("traces-view")), MenuEntry("&Trace Map", main_window.workspace.show_trace_map_view), MenuSeparator(), diff --git a/angrmanagement/ui/toolbars/file_toolbar.py b/angrmanagement/ui/toolbars/file_toolbar.py index f5869c7faa..7f4ee0b333 100644 --- a/angrmanagement/ui/toolbars/file_toolbar.py +++ b/angrmanagement/ui/toolbars/file_toolbar.py @@ -6,6 +6,7 @@ from PySide6.QtGui import QIcon from angrmanagement.config import IMG_LOCATION +from angrmanagement.ui.icons import icon from .toolbar import Toolbar, ToolbarAction @@ -24,7 +25,7 @@ def __init__(self, main_window: MainWindow) -> None: self.actions = [ ToolbarAction( - QIcon(os.path.join(IMG_LOCATION, "toolbar-file-open.ico")), + icon("file-open"), "Open File", "Open a new file for analysis", main_window.open_file_button, @@ -36,7 +37,7 @@ def __init__(self, main_window: MainWindow) -> None: main_window.open_docker_button, ), ToolbarAction( - QIcon(os.path.join(IMG_LOCATION, "toolbar-file-save.png")), + icon("file-save"), "Save", "Save angr database", main_window.save_database, diff --git a/angrmanagement/ui/views/__init__.py b/angrmanagement/ui/views/__init__.py index 0bd835fdc5..eff3ac7cd6 100644 --- a/angrmanagement/ui/views/__init__.py +++ b/angrmanagement/ui/views/__init__.py @@ -7,6 +7,7 @@ from .data_dep_view import DataDepView from .dep_view import DependencyView from .disassembly_view import DisassemblyView +from .explorer_view import ExplorerView from .functions_view import FunctionsView from .hex_view import HexView from .interaction_view import InteractionView @@ -32,6 +33,7 @@ "DataDepView", "DependencyView", "DisassemblyView", + "ExplorerView", "FunctionsView", "HexView", "InteractionView", diff --git a/angrmanagement/ui/views/explorer_view.py b/angrmanagement/ui/views/explorer_view.py new file mode 100644 index 0000000000..c017d463a4 --- /dev/null +++ b/angrmanagement/ui/views/explorer_view.py @@ -0,0 +1,392 @@ +from __future__ import annotations + +from angr.knowledge_plugins.cfg import MemoryDataSort +from PySide6.QtGui import QAction, QStandardItem, QStandardItemModel, Qt +from PySide6.QtWidgets import ( + QHBoxLayout, + QHeaderView, + QLineEdit, + QToolButton, + QTreeView, + QVBoxLayout, +) + +# from angrmanagement.ui.icons import icon +from qtawesome import icon + +from angrmanagement.config import Conf +from angrmanagement.logic import GlobalInfo + +from .view import InstanceView + + +def get_instance(): + workspace = GlobalInfo.main_window.workspace + if workspace: + instance = workspace.main_instance + return instance + return None + + +def get_project(): + instance = get_instance() + if instance: + project = instance.project + if project is not None and not project.am_none: + return project.am_obj + return None + + +class ExplorerTreeModel(QStandardItemModel): + + Headers = ["Function"] + + def hasChildren(self, index): + item: ExplorerTreeItem | None = self.itemFromIndex(index) + if isinstance(item, ExplorerTreeItem): + return item.expandable + return super().hasChildren(index) + + def headerData(self, section, orientation, role): # pylint:disable=unused-argument + if role != Qt.DisplayRole: + return None + if section < len(self.Headers): + return self.Headers[section] + return None + + def refresh(self): + pass + + +class ExplorerTreeItem(QStandardItem): + + expandable: bool = False + + def __init__(self, title, icon=None): + super().__init__(*((icon, title) if icon else (title,))) + self.setEditable(False) + + def expand(self): + pass + + def collapse(self): + while self.rowCount() > 0: + self.removeRow(0) + + def double_clicked(self): + pass + + +class ProjectListItem(ExplorerTreeItem): + + expandable = True + + def __init__(self): + super().__init__("Project") + + def expand(self): + self.appendRows( + [ + LoaderListItem(), + TypesListItem(), + DataListItem(), + FunctionsListItem(), + ] + ) + + +class LoaderListItem(ExplorerTreeItem): + + expandable = True + + def __init__(self): + super().__init__("Loader", icon("mdi.cube-outline")) + + def expand(self): + project = get_project() + if project: + for obj in project.loader.all_objects: + self.appendRow(LoaderObjectItem(obj)) + + +class LoaderObjectItem(ExplorerTreeItem): + + def __init__(self, obj): + self.obj = obj + super().__init__(str(self.obj), icon("mdi.cube-outline", color=self._get_color())) + + def _get_color(self): + if self.obj is self.obj.loader.main_object: + return Qt.green + match self.obj.binary: + case "cle##externs": + return Qt.red + case "cle##kernel": + return Qt.yellow + case "cle##tls": + return Qt.magenta + case _: + return Conf.function_table_color + + def expand(self): + for sec in self.obj.sections: + self.appendRow(LoaderSectionItem(sec)) + + @property + def expandable(self): + return len(self.obj.sections) > 0 + + +class LoaderSectionItem(ExplorerTreeItem): + + def __init__(self, section): + self.section = section + super().__init__(str(self.section), icon("mdi.format-section", color=self._get_color())) + + def _get_color(self): + return Conf.function_table_color + + +class DataListItem(ExplorerTreeItem): + + expandable = True + + def __init__(self): + super().__init__("Data", icon("mdi.data-matrix")) + + def expand(self): + instance = get_instance() + if instance and not instance.cfg.am_none: + for item in sorted(instance.cfg.memory_data.values(), key=lambda i: i.addr): + self.appendRow(DatumItem(item)) + + +class DatumItem(ExplorerTreeItem): + + def __init__(self, item): + self.item = item + project = get_project() + label = "" + if project and item.addr in project.kb.labels: + label = project.kb.labels[item.addr] + ": " + super().__init__(label + str(item), icon(self._get_icon(), color=self._get_color())) + + def _get_icon(self): + match self.item.sort: + case MemoryDataSort.String | MemoryDataSort.UnicodeString: + return "mdi.code-string" + case MemoryDataSort.PointerArray | MemoryDataSort.CodeReference: + return "mdi6.asterisk" + case MemoryDataSort.Integer | MemoryDataSort.FloatingPoint: + return "mdi6.pound-box" + case _: + return "mdi.data-matrix" + + def _get_color(self): + match self.item.sort: + # case MemoryDataSort.Unspecified: + # return + # case MemoryDataSort.Unknown: + # return + case MemoryDataSort.Integer: + return Conf.feature_map_data_color + case MemoryDataSort.PointerArray: + return Conf.feature_map_data_color + case MemoryDataSort.String: + return Conf.feature_map_string_color + case MemoryDataSort.UnicodeString: + return Conf.feature_map_string_color + # case MemoryDataSort.SegmentBoundary: + # return + case MemoryDataSort.CodeReference: + return Conf.function_table_plt_color + case MemoryDataSort.GOTPLTEntry: + return Conf.function_table_plt_color + case MemoryDataSort.ELFHeader: + return Conf.feature_map_data_color + case MemoryDataSort.FloatingPoint: + return Conf.feature_map_data_color + case _: + return Conf.feature_map_unknown_color + + def double_clicked(self): + GlobalInfo.main_window.workspace.jump_to(self.item.addr) + + +class FunctionsListItem(ExplorerTreeItem): + + expandable = True + + def __init__(self): + super().__init__("Functions", icon("mdi.function")) + + def expand(self): + project = get_project() + if project: + for func in project.kb.functions.values(): + self.appendRow(FunctionItem(func)) + + +class FunctionItem(ExplorerTreeItem): + + def __init__(self, function): + self.function = function + super().__init__(function.name, icon("mdi.function", color=self._get_color())) + + def _get_color(self): + func = self.function + if func.is_syscall: + return Conf.function_table_syscall_color + elif func.is_plt: + return Conf.function_table_plt_color + elif func.is_simprocedure: + return Conf.function_table_simprocedure_color + elif func.alignment: + return Conf.function_table_alignment_color + else: + return Conf.function_table_color + + def double_clicked(self): + GlobalInfo.main_window.workspace.on_function_selected(func=self.function) + + +class TypesListItem(ExplorerTreeItem): + + expandable = True + + def __init__(self): + super().__init__("Types", icon("msc.symbol-class")) + + def expand(self): + project = get_project() + if project: + for type_ in project.kb.types.iter_own(): + self.appendRow(TypeItem(type_)) + + +class TypeItem(ExplorerTreeItem): + + def __init__(self, type_): + self.type = type_ + super().__init__(str(self.type), icon("msc.symbol-class")) + + def double_clicked(self): + # GlobalInfo.main_window.workspace.on_type_selected(func=self.type) + pass + + +class ExplorerView(InstanceView): + """ + View displaying functions in the project. + """ + + def __init__(self, *args, **kwargs): + super().__init__("explorer", *args, **kwargs) + + self.base_caption = "Explorer" + + self.instance.cfg.am_subscribe(self.reload) + + self._init_widgets() + + self.width_hint = 375 + self.height_hint = 0 + self.updateGeometry() + + self.function_count = None + + self.reload() + + # + # Public methods + # + + def refresh(self): + self._model.refresh() + + def reload(self): + + self._tree.collapse(self._model.index(0, 0)) + self._tree.expand(self._model.index(0, 0)) + for i in range(self.project_item.rowCount()): + self._tree.expand(self.project_item.child(i).index()) + + # if not self.instance.cfg.am_none: + # self._function_table.function_manager = self.instance.kb.functions + + def subscribe_func_select(self, callback): + """ + Appends the provided function to the list of callbacks to be called when a function is selected in the + functions table. The callback's only parameter is the `angr.knowledge_plugins.functions.function.Function` + :param callback: The callback function to call, which must accept **kwargs + """ + # self._function_table.subscribe_func_select(callback) + + # + # Private methods + # + + def _init_widgets(self): + vlayout = QVBoxLayout() + vlayout.setSpacing(0) + vlayout.setContentsMargins(0, 0, 0, 0) + + hlayout = QHBoxLayout() + hlayout.setSpacing(3) + + tree_options_btn = QToolButton(self) + tree_options_act = QAction(icon("msc.symbol-class"), "Explorer Options") + tree_options_btn.setDefaultAction(tree_options_act) + hlayout.addWidget(tree_options_btn) + + search_box = QLineEdit() + search_box.setClearButtonEnabled(True) + search_box.addAction(icon("fa5s.search", color=Conf.palette_placeholdertext), QLineEdit.LeadingPosition) + search_box.setPlaceholderText("Filter by name...") + hlayout.addWidget(search_box) + + filter_options_btn = QToolButton(self) + filter_options_act = QAction(icon("mdi.filter"), "Filter Options") + filter_options_btn.setDefaultAction(filter_options_act) + hlayout.addWidget(filter_options_btn) + + hlayout.setContentsMargins(3, 3, 3, 3) + vlayout.addLayout(hlayout) + self._tree = QTreeView(self) + self._model = ExplorerTreeModel(self._tree) + self._tree.setModel(self._model) + # self._tree.setFont(QFont(Conf.disasm_font)) + header = self._tree.header() + header.setSectionResizeMode(QHeaderView.ResizeToContents) + header.setVisible(False) + self._tree.expanded.connect(self._on_item_expanded) + self._tree.collapsed.connect(self._on_item_collapsed) + self._tree.doubleClicked.connect(self._on_item_double_clicked) + vlayout.addWidget(self._tree) + self.setLayout(vlayout) + + # self._tree.setStyleSheet("QTreeView { alternate-background-color: yellow;background-color: red; }") + self._tree.setAlternatingRowColors(True) + self.project_item = ProjectListItem() + self._model.appendRow(self.project_item) + + def _on_item_double_clicked(self, index): + """ + Handle item double-click event. + """ + item = self._model.itemFromIndex(index) + item.double_clicked() + + def _on_item_expanded(self, index): + """ + Handle item expansion. + """ + item = self._model.itemFromIndex(index) + item.expand() + + def _on_item_collapsed(self, index): + """ + Handle item collapse. + """ + item = self._model.itemFromIndex(index) + item.collapse() diff --git a/angrmanagement/ui/workspace.py b/angrmanagement/ui/workspace.py index 22ab54bed5..6a9e48cc7e 100644 --- a/angrmanagement/ui/workspace.py +++ b/angrmanagement/ui/workspace.py @@ -58,6 +58,7 @@ DataDepView, DependencyView, DisassemblyView, + ExplorerView, FunctionsView, HexView, InteractionView, @@ -118,6 +119,7 @@ def __init__(self, main_window: MainWindow) -> None: HexView(self, "center", self._main_instance), CodeView(self, "center", self._main_instance), FunctionsView(self, "left", self._main_instance), + ExplorerView(self, "left", self._main_instance), ] if Conf.has_operation_mango: self.default_tabs.append(DependencyView(self, "center", self._main_instance)) @@ -822,6 +824,9 @@ def show_types_view(self) -> None: def show_functions_view(self) -> None: self.show_view("functions", FunctionsView, position="left") + def show_explorer_view(self) -> None: + self.show_view("explorer", ExplorerView, position="left") + def show_traces_view(self) -> None: self.show_view("traces", TracesView)