use anyhow::{bail, Result}; use crossterm::event::{Event, KeyCode, KeyEventKind, KeyModifiers, MouseButton, MouseEventKind}; use objdiff_core::{ diff::{ display::{display_diff, DiffText, HighlightKind}, ObjDiff, ObjInsDiffKind, ObjSymbolDiff, }, obj::{ObjInfo, ObjSectionKind, ObjSymbol, SymbolRef}, }; use ratatui::{ prelude::*, widgets::{Block, Borders, Clear, Paragraph, Scrollbar, ScrollbarOrientation, ScrollbarState}, Frame, }; use super::{EventControlFlow, EventResult, UiView}; use crate::cmd::diff::AppState; #[allow(dead_code)] #[derive(Default)] pub struct FunctionDiffUi { pub symbol_name: String, pub left_highlight: HighlightKind, pub right_highlight: HighlightKind, pub scroll_x: usize, pub scroll_state_x: ScrollbarState, pub scroll_y: usize, pub scroll_state_y: ScrollbarState, pub per_page: usize, pub num_rows: usize, pub left_sym: Option, pub right_sym: Option, pub prev_sym: Option, pub open_options: bool, pub three_way: bool, } impl UiView for FunctionDiffUi { fn draw(&mut self, state: &AppState, f: &mut Frame, result: &mut EventResult) { let chunks = Layout::vertical([Constraint::Length(1), Constraint::Fill(1)]).split(f.area()); let header_chunks = Layout::horizontal([ Constraint::Fill(1), Constraint::Length(3), Constraint::Fill(1), Constraint::Length(2), ]) .split(chunks[0]); let content_chunks = if self.three_way { Layout::horizontal([ Constraint::Fill(1), Constraint::Length(3), Constraint::Fill(1), Constraint::Length(3), Constraint::Fill(1), Constraint::Length(2), ]) .split(chunks[1]) } else { Layout::horizontal([ Constraint::Fill(1), Constraint::Length(3), Constraint::Fill(1), Constraint::Length(2), ]) .split(chunks[1]) }; self.per_page = chunks[1].height.saturating_sub(2) as usize; let max_scroll_y = self.num_rows.saturating_sub(self.per_page); if self.scroll_y > max_scroll_y { self.scroll_y = max_scroll_y; } self.scroll_state_y = self.scroll_state_y.content_length(max_scroll_y).position(self.scroll_y); let mut line_l = Line::default(); line_l .spans .push(Span::styled(self.symbol_name.clone(), Style::new().fg(Color::White).bold())); f.render_widget(line_l, header_chunks[0]); let mut line_r = Line::default(); if let Some(percent) = get_symbol(state.right_obj.as_ref(), self.right_sym).and_then(|(_, d)| d.match_percent) { line_r.spans.push(Span::styled( format!("{:.2}% ", percent), Style::new().fg(match_percent_color(percent)), )); } let reload_time = state .reload_time .as_ref() .and_then(|t| t.format(&state.time_format).ok()) .unwrap_or_else(|| "N/A".to_string()); line_r.spans.push(Span::styled( format!("Last reload: {}", reload_time), Style::new().fg(Color::White), )); line_r.spans.push(Span::styled( format!(" ({} jobs)", state.jobs.jobs.len()), Style::new().fg(Color::LightYellow), )); f.render_widget(line_r, header_chunks[2]); let mut left_text = None; let mut left_highlight = None; let mut max_width = 0; if let Some((symbol, symbol_diff)) = get_symbol(state.left_obj.as_ref(), self.left_sym) { let mut text = Text::default(); let rect = content_chunks[0].inner(Margin::new(0, 1)); left_highlight = self.print_sym( &mut text, symbol, symbol_diff, rect, &self.left_highlight, result, false, ); max_width = max_width.max(text.width()); left_text = Some(text); } let mut right_text = None; let mut right_highlight = None; let mut margin_text = None; if let Some((symbol, symbol_diff)) = get_symbol(state.right_obj.as_ref(), self.right_sym) { let mut text = Text::default(); let rect = content_chunks[2].inner(Margin::new(0, 1)); right_highlight = self.print_sym( &mut text, symbol, symbol_diff, rect, &self.right_highlight, result, false, ); max_width = max_width.max(text.width()); right_text = Some(text); // Render margin let mut text = Text::default(); let rect = content_chunks[1].inner(Margin::new(1, 1)); self.print_margin(&mut text, symbol_diff, rect); margin_text = Some(text); } let mut prev_text = None; let mut prev_margin_text = None; if self.three_way { if let Some((symbol, symbol_diff)) = get_symbol(state.prev_obj.as_ref(), self.prev_sym) { let mut text = Text::default(); let rect = content_chunks[4].inner(Margin::new(0, 1)); self.print_sym( &mut text, symbol, symbol_diff, rect, &self.right_highlight, result, true, ); max_width = max_width.max(text.width()); prev_text = Some(text); // Render margin let mut text = Text::default(); let rect = content_chunks[3].inner(Margin::new(1, 1)); self.print_margin(&mut text, symbol_diff, rect); prev_margin_text = Some(text); } } let max_scroll_x = max_width.saturating_sub(content_chunks[0].width.min(content_chunks[2].width) as usize); if self.scroll_x > max_scroll_x { self.scroll_x = max_scroll_x; } self.scroll_state_x = self.scroll_state_x.content_length(max_scroll_x).position(self.scroll_x); if let Some(text) = left_text { // Render left column f.render_widget( Paragraph::new(text) .block( Block::new() .borders(Borders::TOP) .border_style(Style::new().fg(Color::Gray)) .title_style(Style::new().bold()) .title("TARGET"), ) .scroll((0, self.scroll_x as u16)), content_chunks[0], ); } if let Some(text) = margin_text { f.render_widget(text, content_chunks[1].inner(Margin::new(1, 1))); } if let Some(text) = right_text { f.render_widget( Paragraph::new(text) .block( Block::new() .borders(Borders::TOP) .border_style(Style::new().fg(Color::Gray)) .title_style(Style::new().bold()) .title("CURRENT"), ) .scroll((0, self.scroll_x as u16)), content_chunks[2], ); } if self.three_way { if let Some(text) = prev_margin_text { f.render_widget(text, content_chunks[3].inner(Margin::new(1, 1))); } let block = Block::new() .borders(Borders::TOP) .border_style(Style::new().fg(Color::Gray)) .title_style(Style::new().bold()) .title("SAVED"); if let Some(text) = prev_text { f.render_widget( Paragraph::new(text).block(block.clone()).scroll((0, self.scroll_x as u16)), content_chunks[4], ); } else { f.render_widget(block, content_chunks[4]); } } // Render scrollbars f.render_stateful_widget( Scrollbar::new(ScrollbarOrientation::VerticalRight).begin_symbol(None).end_symbol(None), chunks[1].inner(Margin::new(0, 1)), &mut self.scroll_state_y, ); f.render_stateful_widget( Scrollbar::new(ScrollbarOrientation::HorizontalBottom).thumb_symbol("■"), content_chunks[0], &mut self.scroll_state_x, ); f.render_stateful_widget( Scrollbar::new(ScrollbarOrientation::HorizontalBottom).thumb_symbol("■"), content_chunks[2], &mut self.scroll_state_x, ); if self.three_way { f.render_stateful_widget( Scrollbar::new(ScrollbarOrientation::HorizontalBottom).thumb_symbol("■"), content_chunks[4], &mut self.scroll_state_x, ); } if let Some(new_highlight) = left_highlight { if new_highlight == self.left_highlight { if self.left_highlight != self.right_highlight { self.right_highlight = self.left_highlight.clone(); } else { self.left_highlight = HighlightKind::None; self.right_highlight = HighlightKind::None; } } else { self.left_highlight = new_highlight; } result.redraw = true; } else if let Some(new_highlight) = right_highlight { if new_highlight == self.right_highlight { if self.left_highlight != self.right_highlight { self.left_highlight = self.right_highlight.clone(); } else { self.left_highlight = HighlightKind::None; self.right_highlight = HighlightKind::None; } } else { self.right_highlight = new_highlight; } result.redraw = true; } if self.open_options { self.draw_options(f, result); } } fn handle_event(&mut self, state: &mut AppState, event: Event) -> EventControlFlow { let mut result = EventResult::default(); match event { Event::Key(event) if matches!(event.kind, KeyEventKind::Press | KeyEventKind::Repeat) => { match event.code { // Quit KeyCode::Esc | KeyCode::Char('q') => return EventControlFlow::Break, // Page up KeyCode::PageUp => { self.page_up(false); result.redraw = true; } // Page up (shift + space) KeyCode::Char(' ') if event.modifiers.contains(KeyModifiers::SHIFT) => { self.page_up(false); result.redraw = true; } // Page down KeyCode::Char(' ') | KeyCode::PageDown => { self.page_down(false); result.redraw = true; } // Page down (ctrl + f) KeyCode::Char('f') if event.modifiers.contains(KeyModifiers::CONTROL) => { self.page_down(false); result.redraw = true; } // Page up (ctrl + b) KeyCode::Char('b') if event.modifiers.contains(KeyModifiers::CONTROL) => { self.page_up(false); result.redraw = true; } // Half page down (ctrl + d) KeyCode::Char('d') if event.modifiers.contains(KeyModifiers::CONTROL) => { self.page_down(true); result.redraw = true; } // Half page up (ctrl + u) KeyCode::Char('u') if event.modifiers.contains(KeyModifiers::CONTROL) => { self.page_up(true); result.redraw = true; } // Scroll down KeyCode::Down | KeyCode::Char('j') => { self.scroll_y += 1; result.redraw = true; } // Scroll up KeyCode::Up | KeyCode::Char('k') => { self.scroll_y = self.scroll_y.saturating_sub(1); result.redraw = true; } // Scroll to start KeyCode::Char('g') => { self.scroll_y = 0; result.redraw = true; } // Scroll to end KeyCode::Char('G') => { self.scroll_y = self.num_rows; result.redraw = true; } // Reload KeyCode::Char('r') => { result.redraw = true; return EventControlFlow::Reload; } // Scroll right KeyCode::Right | KeyCode::Char('l') => { self.scroll_x += 1; result.redraw = true; } // Scroll left KeyCode::Left | KeyCode::Char('h') => { self.scroll_x = self.scroll_x.saturating_sub(1); result.redraw = true; } // Toggle relax relocation diffs KeyCode::Char('x') => { state.diff_obj_config.relax_reloc_diffs = !state.diff_obj_config.relax_reloc_diffs; result.redraw = true; return EventControlFlow::Reload; } // Toggle three-way diff KeyCode::Char('3') => { self.three_way = !self.three_way; result.redraw = true; } // Toggle options KeyCode::Char('o') => { self.open_options = !self.open_options; result.redraw = true; } _ => {} } } Event::Mouse(event) => match event.kind { MouseEventKind::ScrollDown => { self.scroll_y += 3; result.redraw = true; } MouseEventKind::ScrollUp => { self.scroll_y = self.scroll_y.saturating_sub(3); result.redraw = true; } MouseEventKind::ScrollRight => { self.scroll_x += 3; result.redraw = true; } MouseEventKind::ScrollLeft => { self.scroll_x = self.scroll_x.saturating_sub(3); result.redraw = true; } MouseEventKind::Down(MouseButton::Left) => { result.click_xy = Some((event.column, event.row)); result.redraw = true; } _ => {} }, Event::Resize(_, _) => { result.redraw = true; } _ => {} } EventControlFlow::Continue(result) } fn reload(&mut self, state: &AppState) -> Result<()> { let left_sym = state.left_obj.as_ref().and_then(|(o, _)| find_function(o, &self.symbol_name)); let right_sym = state.right_obj.as_ref().and_then(|(o, _)| find_function(o, &self.symbol_name)); let prev_sym = state.prev_obj.as_ref().and_then(|(o, _)| find_function(o, &self.symbol_name)); self.num_rows = match ( get_symbol(state.left_obj.as_ref(), left_sym), get_symbol(state.right_obj.as_ref(), right_sym), ) { (Some((_l, ld)), Some((_r, rd))) => ld.instructions.len().max(rd.instructions.len()), (Some((_l, ld)), None) => ld.instructions.len(), (None, Some((_r, rd))) => rd.instructions.len(), (None, None) => bail!("Symbol not found: {}", self.symbol_name), }; self.left_sym = left_sym; self.right_sym = right_sym; self.prev_sym = prev_sym; Ok(()) } } impl FunctionDiffUi { pub fn draw_options(&mut self, f: &mut Frame, _result: &mut EventResult) { let percent_x = 50; let percent_y = 50; let popup_rect = Layout::vertical([ Constraint::Percentage((100 - percent_y) / 2), Constraint::Percentage(percent_y), Constraint::Percentage((100 - percent_y) / 2), ]) .split(f.area())[1]; let popup_rect = Layout::horizontal([ Constraint::Percentage((100 - percent_x) / 2), Constraint::Percentage(percent_x), Constraint::Percentage((100 - percent_x) / 2), ]) .split(popup_rect)[1]; let popup = Block::default() .borders(Borders::ALL) .title("Options") .title_style(Style::default().fg(Color::White).bg(Color::Black)); f.render_widget(Clear, popup_rect); f.render_widget(popup, popup_rect); } fn page_up(&mut self, half: bool) { self.scroll_y = self.scroll_y.saturating_sub(self.per_page / if half { 2 } else { 1 }); } fn page_down(&mut self, half: bool) { self.scroll_y += self.per_page / if half { 2 } else { 1 }; } #[allow(clippy::too_many_arguments)] fn print_sym( &self, out: &mut Text<'static>, symbol: &ObjSymbol, symbol_diff: &ObjSymbolDiff, rect: Rect, highlight: &HighlightKind, result: &EventResult, only_changed: bool, ) -> Option { let base_addr = symbol.address; let mut new_highlight = None; for (y, ins_diff) in symbol_diff .instructions .iter() .skip(self.scroll_y) .take(rect.height as usize) .enumerate() { if only_changed && ins_diff.kind == ObjInsDiffKind::None { out.lines.push(Line::default()); continue; } let mut sx = rect.x; let sy = rect.y + y as u16; let mut line = Line::default(); display_diff(ins_diff, base_addr, |text| -> Result<()> { let label_text; let mut base_color = match ins_diff.kind { ObjInsDiffKind::None | ObjInsDiffKind::OpMismatch | ObjInsDiffKind::ArgMismatch => Color::Gray, ObjInsDiffKind::Replace => Color::Cyan, ObjInsDiffKind::Delete => Color::Red, ObjInsDiffKind::Insert => Color::Green, }; let mut pad_to = 0; match text { DiffText::Basic(text) => { label_text = text.to_string(); } DiffText::BasicColor(s, idx) => { label_text = s.to_string(); base_color = COLOR_ROTATION[idx % COLOR_ROTATION.len()]; } DiffText::Line(num) => { label_text = format!("{num} "); base_color = Color::DarkGray; pad_to = 5; } DiffText::Address(addr) => { label_text = format!("{:x}:", addr); pad_to = 5; } DiffText::Opcode(mnemonic, _op) => { label_text = mnemonic.to_string(); if ins_diff.kind == ObjInsDiffKind::OpMismatch { base_color = Color::Blue; } pad_to = 8; } DiffText::Argument(arg, diff) => { label_text = arg.to_string(); if let Some(diff) = diff { base_color = COLOR_ROTATION[diff.idx % COLOR_ROTATION.len()] } } DiffText::BranchDest(addr, diff) => { label_text = format!("{addr:x}"); if let Some(diff) = diff { base_color = COLOR_ROTATION[diff.idx % COLOR_ROTATION.len()] } } DiffText::Symbol(sym, diff) => { let name = sym.demangled_name.as_ref().unwrap_or(&sym.name); label_text = name.clone(); if let Some(diff) = diff { base_color = COLOR_ROTATION[diff.idx % COLOR_ROTATION.len()] } else { base_color = Color::White; } } DiffText::Spacing(n) => { line.spans.push(Span::raw(" ".repeat(n))); sx += n as u16; return Ok(()); } DiffText::Eol => { return Ok(()); } } let len = label_text.len(); let highlighted = *highlight == text; if let Some((cx, cy)) = result.click_xy { if cx >= sx && cx < sx + len as u16 && cy == sy { new_highlight = Some(text.into()); } } let mut style = Style::new().fg(base_color); if highlighted { style = style.bg(Color::DarkGray); } line.spans.push(Span::styled(label_text, style)); sx += len as u16; if pad_to > len { let pad = (pad_to - len) as u16; line.spans.push(Span::raw(" ".repeat(pad as usize))); sx += pad; } Ok(()) }) .unwrap(); out.lines.push(line); } new_highlight } fn print_margin(&self, out: &mut Text, symbol: &ObjSymbolDiff, rect: Rect) { for ins_diff in symbol.instructions.iter().skip(self.scroll_y).take(rect.height as usize) { if ins_diff.kind != ObjInsDiffKind::None { out.lines.push(Line::raw(match ins_diff.kind { ObjInsDiffKind::Delete => "<", ObjInsDiffKind::Insert => ">", _ => "|", })); } else { out.lines.push(Line::raw(" ")); } } } } pub const COLOR_ROTATION: [Color; 7] = [ Color::Magenta, Color::Cyan, Color::Green, Color::Red, Color::Yellow, Color::Blue, Color::Green, ]; pub fn match_percent_color(match_percent: f32) -> Color { if match_percent == 100.0 { Color::Green } else if match_percent >= 50.0 { Color::LightBlue } else { Color::LightRed } } #[inline] fn get_symbol( obj: Option<&(ObjInfo, ObjDiff)>, sym: Option, ) -> Option<(&ObjSymbol, &ObjSymbolDiff)> { let (obj, diff) = obj?; let sym = sym?; Some((obj.section_symbol(sym).1, diff.symbol_diff(sym))) } fn find_function(obj: &ObjInfo, name: &str) -> Option { for (section_idx, section) in obj.sections.iter().enumerate() { if section.kind != ObjSectionKind::Code { continue; } for (symbol_idx, symbol) in section.symbols.iter().enumerate() { if symbol.name == name { return Some(SymbolRef { section_idx, symbol_idx }); } } } None }