mirror of https://github.com/encounter/objdiff.git
Experimental objdiff-cli diff auto-rebuild
This commit is contained in:
parent
10b2a9c129
commit
526e031251
|
@ -1398,6 +1398,14 @@ dependencies = [
|
||||||
"simd-adler32",
|
"simd-adler32",
|
||||||
]
|
]
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "file-id"
|
||||||
|
version = "0.2.1"
|
||||||
|
source = "git+https://github.com/notify-rs/notify?rev=128bf6230c03d39dbb7f301ff7b20e594e34c3a2#128bf6230c03d39dbb7f301ff7b20e594e34c3a2"
|
||||||
|
dependencies = [
|
||||||
|
"windows-sys 0.52.0",
|
||||||
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "filetime"
|
name = "filetime"
|
||||||
version = "0.2.25"
|
version = "0.2.25"
|
||||||
|
@ -2726,6 +2734,18 @@ dependencies = [
|
||||||
"windows-sys 0.52.0",
|
"windows-sys 0.52.0",
|
||||||
]
|
]
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "notify-debouncer-full"
|
||||||
|
version = "0.4.0"
|
||||||
|
source = "git+https://github.com/notify-rs/notify?rev=128bf6230c03d39dbb7f301ff7b20e594e34c3a2#128bf6230c03d39dbb7f301ff7b20e594e34c3a2"
|
||||||
|
dependencies = [
|
||||||
|
"file-id",
|
||||||
|
"log",
|
||||||
|
"notify",
|
||||||
|
"notify-types",
|
||||||
|
"walkdir",
|
||||||
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "notify-types"
|
name = "notify-types"
|
||||||
version = "1.0.0"
|
version = "1.0.0"
|
||||||
|
@ -3071,23 +3091,32 @@ dependencies = [
|
||||||
"log",
|
"log",
|
||||||
"memmap2",
|
"memmap2",
|
||||||
"msvc-demangler",
|
"msvc-demangler",
|
||||||
|
"notify",
|
||||||
|
"notify-debouncer-full",
|
||||||
"num-traits",
|
"num-traits",
|
||||||
"object",
|
"object",
|
||||||
|
"path-slash",
|
||||||
"pbjson",
|
"pbjson",
|
||||||
"pbjson-build",
|
"pbjson-build",
|
||||||
"ppc750cl",
|
"ppc750cl",
|
||||||
"prost",
|
"prost",
|
||||||
"prost-build",
|
"prost-build",
|
||||||
"rabbitizer",
|
"rabbitizer",
|
||||||
|
"reqwest",
|
||||||
|
"self_update",
|
||||||
"semver",
|
"semver",
|
||||||
"serde",
|
"serde",
|
||||||
"serde_json",
|
"serde_json",
|
||||||
"serde_yaml",
|
"serde_yaml",
|
||||||
|
"shell-escape",
|
||||||
"similar",
|
"similar",
|
||||||
"strum",
|
"strum",
|
||||||
|
"tempfile",
|
||||||
|
"time",
|
||||||
"tsify-next",
|
"tsify-next",
|
||||||
"unarm",
|
"unarm",
|
||||||
"wasm-bindgen",
|
"wasm-bindgen",
|
||||||
|
"winapi",
|
||||||
"yaxpeax-arch",
|
"yaxpeax-arch",
|
||||||
"yaxpeax-arm",
|
"yaxpeax-arm",
|
||||||
]
|
]
|
||||||
|
@ -3113,24 +3142,19 @@ dependencies = [
|
||||||
"font-kit",
|
"font-kit",
|
||||||
"globset",
|
"globset",
|
||||||
"log",
|
"log",
|
||||||
"notify",
|
|
||||||
"objdiff-core",
|
"objdiff-core",
|
||||||
"open",
|
"open",
|
||||||
"path-slash",
|
|
||||||
"png",
|
"png",
|
||||||
"pollster 0.4.0",
|
"pollster 0.4.0",
|
||||||
"regex",
|
"regex",
|
||||||
"reqwest",
|
|
||||||
"rfd",
|
"rfd",
|
||||||
"rlwinmdec",
|
"rlwinmdec",
|
||||||
"ron",
|
"ron",
|
||||||
"self_update",
|
|
||||||
"serde",
|
"serde",
|
||||||
"serde_json",
|
"serde_json",
|
||||||
"shell-escape",
|
"shell-escape",
|
||||||
"strum",
|
"strum",
|
||||||
"tauri-winres",
|
"tauri-winres",
|
||||||
"tempfile",
|
|
||||||
"time",
|
"time",
|
||||||
"tracing-subscriber",
|
"tracing-subscriber",
|
||||||
"tracing-wasm",
|
"tracing-wasm",
|
||||||
|
|
|
@ -1,42 +1,50 @@
|
||||||
use std::{
|
use std::{
|
||||||
fs,
|
fs,
|
||||||
io::stdout,
|
io::stdout,
|
||||||
|
mem,
|
||||||
path::{Path, PathBuf},
|
path::{Path, PathBuf},
|
||||||
str::FromStr,
|
str::FromStr,
|
||||||
|
sync::{
|
||||||
|
atomic::{AtomicBool, Ordering},
|
||||||
|
Arc,
|
||||||
|
},
|
||||||
|
task::{Wake, Waker},
|
||||||
|
time::Duration,
|
||||||
};
|
};
|
||||||
|
|
||||||
use anyhow::{bail, Context, Result};
|
use anyhow::{bail, Context, Result};
|
||||||
use argp::FromArgs;
|
use argp::FromArgs;
|
||||||
use crossterm::{
|
use crossterm::{
|
||||||
event,
|
event,
|
||||||
event::{
|
event::{DisableMouseCapture, EnableMouseCapture},
|
||||||
DisableMouseCapture, EnableMouseCapture, Event, KeyCode, KeyEventKind, MouseButton,
|
|
||||||
MouseEventKind,
|
|
||||||
},
|
|
||||||
terminal::{
|
terminal::{
|
||||||
disable_raw_mode, enable_raw_mode, EnterAlternateScreen, LeaveAlternateScreen, SetTitle,
|
disable_raw_mode, enable_raw_mode, EnterAlternateScreen, LeaveAlternateScreen, SetTitle,
|
||||||
},
|
},
|
||||||
};
|
};
|
||||||
use event::KeyModifiers;
|
|
||||||
use objdiff_core::{
|
use objdiff_core::{
|
||||||
bindings::diff::DiffResult,
|
bindings::diff::DiffResult,
|
||||||
config::{ProjectConfig, ProjectObject},
|
build::{
|
||||||
|
watcher::{create_watcher, Watcher},
|
||||||
|
BuildConfig,
|
||||||
|
},
|
||||||
|
config::{build_globset, default_watch_patterns, ProjectConfig, ProjectObject},
|
||||||
diff,
|
diff,
|
||||||
diff::{
|
diff::ObjDiff,
|
||||||
display::{display_diff, DiffText, HighlightKind},
|
jobs::{
|
||||||
DiffObjsResult, ObjDiff, ObjInsDiffKind, ObjSymbolDiff,
|
objdiff::{start_build, ObjDiffConfig},
|
||||||
|
Job, JobQueue, JobResult,
|
||||||
},
|
},
|
||||||
obj,
|
obj,
|
||||||
obj::{ObjInfo, ObjSectionKind, ObjSymbol, SymbolRef},
|
obj::ObjInfo,
|
||||||
};
|
|
||||||
use ratatui::{
|
|
||||||
prelude::*,
|
|
||||||
widgets::{Block, Borders, Clear, Paragraph, Scrollbar, ScrollbarOrientation, ScrollbarState},
|
|
||||||
};
|
};
|
||||||
|
use ratatui::prelude::*;
|
||||||
|
|
||||||
use crate::util::{
|
use crate::{
|
||||||
|
util::{
|
||||||
output::{write_output, OutputFormat},
|
output::{write_output, OutputFormat},
|
||||||
term::crossterm_panic_handler,
|
term::crossterm_panic_handler,
|
||||||
|
},
|
||||||
|
views::{function_diff::FunctionDiffUi, EventControlFlow, EventResult, UiView},
|
||||||
};
|
};
|
||||||
|
|
||||||
#[derive(FromArgs, PartialEq, Debug)]
|
#[derive(FromArgs, PartialEq, Debug)]
|
||||||
|
@ -209,6 +217,94 @@ fn run_oneshot(
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pub struct AppState {
|
||||||
|
pub jobs: JobQueue,
|
||||||
|
pub waker: Arc<TermWaker>,
|
||||||
|
pub project_dir: Option<PathBuf>,
|
||||||
|
pub project_config: Option<ProjectConfig>,
|
||||||
|
pub target_path: Option<PathBuf>,
|
||||||
|
pub base_path: Option<PathBuf>,
|
||||||
|
pub left_obj: Option<(ObjInfo, ObjDiff)>,
|
||||||
|
pub right_obj: Option<(ObjInfo, ObjDiff)>,
|
||||||
|
pub prev_obj: Option<(ObjInfo, ObjDiff)>,
|
||||||
|
pub reload_time: Option<time::OffsetDateTime>,
|
||||||
|
pub time_format: Vec<time::format_description::FormatItem<'static>>,
|
||||||
|
pub relax_reloc_diffs: bool,
|
||||||
|
pub watcher: Option<Watcher>,
|
||||||
|
pub modified: Arc<AtomicBool>,
|
||||||
|
}
|
||||||
|
|
||||||
|
fn create_objdiff_config(state: &AppState) -> ObjDiffConfig {
|
||||||
|
ObjDiffConfig {
|
||||||
|
build_config: BuildConfig {
|
||||||
|
project_dir: state.project_dir.clone(),
|
||||||
|
custom_make: state
|
||||||
|
.project_config
|
||||||
|
.as_ref()
|
||||||
|
.and_then(|c| c.custom_make.as_ref())
|
||||||
|
.cloned(),
|
||||||
|
custom_args: state
|
||||||
|
.project_config
|
||||||
|
.as_ref()
|
||||||
|
.and_then(|c| c.custom_args.as_ref())
|
||||||
|
.cloned(),
|
||||||
|
selected_wsl_distro: None,
|
||||||
|
},
|
||||||
|
build_base: state.project_config.as_ref().is_some_and(|p| p.build_base.unwrap_or(true)),
|
||||||
|
build_target: state
|
||||||
|
.project_config
|
||||||
|
.as_ref()
|
||||||
|
.is_some_and(|p| p.build_target.unwrap_or(false)),
|
||||||
|
target_path: state.target_path.clone(),
|
||||||
|
base_path: state.base_path.clone(),
|
||||||
|
diff_obj_config: diff::DiffObjConfig {
|
||||||
|
relax_reloc_diffs: state.relax_reloc_diffs,
|
||||||
|
..Default::default() // TODO
|
||||||
|
},
|
||||||
|
symbol_mappings: Default::default(),
|
||||||
|
selecting_left: None,
|
||||||
|
selecting_right: None,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl AppState {
|
||||||
|
fn reload(&mut self) -> Result<()> {
|
||||||
|
let config = create_objdiff_config(self);
|
||||||
|
self.jobs.push_once(Job::ObjDiff, || start_build(Waker::from(self.waker.clone()), config));
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
||||||
|
fn check_jobs(&mut self) -> Result<bool> {
|
||||||
|
let mut redraw = false;
|
||||||
|
self.jobs.collect_results();
|
||||||
|
for result in mem::take(&mut self.jobs.results) {
|
||||||
|
match result {
|
||||||
|
JobResult::None => unreachable!("Unexpected JobResult::None"),
|
||||||
|
JobResult::ObjDiff(result) => {
|
||||||
|
let result = result.unwrap();
|
||||||
|
self.left_obj = result.first_obj;
|
||||||
|
self.right_obj = result.second_obj;
|
||||||
|
self.reload_time = Some(result.time);
|
||||||
|
redraw = true;
|
||||||
|
}
|
||||||
|
JobResult::CheckUpdate(_) => todo!("CheckUpdate"),
|
||||||
|
JobResult::Update(_) => todo!("Update"),
|
||||||
|
JobResult::CreateScratch(_) => todo!("CreateScratch"),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
Ok(redraw)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(Default)]
|
||||||
|
pub struct TermWaker(pub AtomicBool);
|
||||||
|
|
||||||
|
impl Wake for TermWaker {
|
||||||
|
fn wake(self: Arc<Self>) { self.0.store(true, Ordering::Relaxed); }
|
||||||
|
|
||||||
|
fn wake_by_ref(self: &Arc<Self>) { self.0.store(true, Ordering::Relaxed); }
|
||||||
|
}
|
||||||
|
|
||||||
fn run_interactive(
|
fn run_interactive(
|
||||||
args: Args,
|
args: Args,
|
||||||
target_path: Option<PathBuf>,
|
target_path: Option<PathBuf>,
|
||||||
|
@ -218,32 +314,38 @@ fn run_interactive(
|
||||||
let Some(symbol_name) = &args.symbol else { bail!("Interactive mode requires a symbol name") };
|
let Some(symbol_name) = &args.symbol else { bail!("Interactive mode requires a symbol name") };
|
||||||
let time_format = time::format_description::parse_borrowed::<2>("[hour]:[minute]:[second]")
|
let time_format = time::format_description::parse_borrowed::<2>("[hour]:[minute]:[second]")
|
||||||
.context("Failed to parse time format")?;
|
.context("Failed to parse time format")?;
|
||||||
let mut state = Box::new(FunctionDiffUi {
|
let mut state = AppState {
|
||||||
relax_reloc_diffs: args.relax_reloc_diffs,
|
jobs: Default::default(),
|
||||||
left_highlight: HighlightKind::None,
|
waker: Default::default(),
|
||||||
right_highlight: HighlightKind::None,
|
project_dir: args.project.clone(),
|
||||||
scroll_x: 0,
|
project_config,
|
||||||
scroll_state_x: ScrollbarState::default(),
|
|
||||||
scroll_y: 0,
|
|
||||||
scroll_state_y: ScrollbarState::default(),
|
|
||||||
per_page: 0,
|
|
||||||
num_rows: 0,
|
|
||||||
symbol_name: symbol_name.clone(),
|
|
||||||
target_path,
|
target_path,
|
||||||
base_path,
|
base_path,
|
||||||
project_config,
|
|
||||||
left_obj: None,
|
left_obj: None,
|
||||||
right_obj: None,
|
right_obj: None,
|
||||||
prev_obj: None,
|
prev_obj: None,
|
||||||
diff_result: DiffObjsResult::default(),
|
|
||||||
left_sym: None,
|
|
||||||
right_sym: None,
|
|
||||||
prev_sym: None,
|
|
||||||
reload_time: None,
|
reload_time: None,
|
||||||
time_format,
|
time_format,
|
||||||
open_options: false,
|
relax_reloc_diffs: args.relax_reloc_diffs,
|
||||||
three_way: false,
|
watcher: None,
|
||||||
});
|
modified: Default::default(),
|
||||||
|
};
|
||||||
|
if let Some(project_dir) = &state.project_dir {
|
||||||
|
let watch_patterns = state
|
||||||
|
.project_config
|
||||||
|
.as_ref()
|
||||||
|
.and_then(|c| c.watch_patterns.as_ref())
|
||||||
|
.cloned()
|
||||||
|
.unwrap_or_else(default_watch_patterns);
|
||||||
|
state.watcher = Some(create_watcher(
|
||||||
|
state.modified.clone(),
|
||||||
|
project_dir,
|
||||||
|
build_globset(&watch_patterns)?,
|
||||||
|
Waker::from(state.waker.clone()),
|
||||||
|
)?);
|
||||||
|
}
|
||||||
|
let mut view: Box<dyn UiView> =
|
||||||
|
Box::new(FunctionDiffUi { symbol_name: symbol_name.clone(), ..Default::default() });
|
||||||
state.reload()?;
|
state.reload()?;
|
||||||
|
|
||||||
crossterm_panic_handler();
|
crossterm_panic_handler();
|
||||||
|
@ -257,16 +359,12 @@ fn run_interactive(
|
||||||
let backend = CrosstermBackend::new(stdout());
|
let backend = CrosstermBackend::new(stdout());
|
||||||
let mut terminal = Terminal::new(backend)?;
|
let mut terminal = Terminal::new(backend)?;
|
||||||
|
|
||||||
'outer: loop {
|
|
||||||
let mut result = EventResult { redraw: true, ..Default::default() };
|
let mut result = EventResult { redraw: true, ..Default::default() };
|
||||||
loop {
|
'outer: loop {
|
||||||
if result.redraw {
|
if result.redraw {
|
||||||
terminal.draw(|f| loop {
|
terminal.draw(|f| loop {
|
||||||
result.redraw = false;
|
result.redraw = false;
|
||||||
state.draw(f, &mut result);
|
view.draw(&state, f, &mut result);
|
||||||
if state.open_options {
|
|
||||||
state.draw_options(f, &mut result);
|
|
||||||
}
|
|
||||||
result.click_xy = None;
|
result.click_xy = None;
|
||||||
if !result.redraw {
|
if !result.redraw {
|
||||||
break;
|
break;
|
||||||
|
@ -275,13 +373,29 @@ fn run_interactive(
|
||||||
f.buffer_mut().reset();
|
f.buffer_mut().reset();
|
||||||
})?;
|
})?;
|
||||||
}
|
}
|
||||||
match state.handle_event(event::read()?) {
|
loop {
|
||||||
|
if event::poll(Duration::from_millis(100))? {
|
||||||
|
match view.handle_event(&mut state, event::read()?) {
|
||||||
EventControlFlow::Break => break 'outer,
|
EventControlFlow::Break => break 'outer,
|
||||||
EventControlFlow::Continue(r) => result = r,
|
EventControlFlow::Continue(r) => result = r,
|
||||||
EventControlFlow::Reload => break,
|
EventControlFlow::Reload => {
|
||||||
}
|
|
||||||
}
|
|
||||||
state.reload()?;
|
state.reload()?;
|
||||||
|
result.redraw = true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
} else if state.waker.0.swap(false, Ordering::Relaxed) {
|
||||||
|
if state.modified.swap(false, Ordering::Relaxed) {
|
||||||
|
state.reload()?;
|
||||||
|
}
|
||||||
|
result.redraw = true;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if state.check_jobs()? {
|
||||||
|
result.redraw = true;
|
||||||
|
view.reload(&state)?;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Reset terminal
|
// Reset terminal
|
||||||
|
@ -290,680 +404,3 @@ fn run_interactive(
|
||||||
terminal.show_cursor()?;
|
terminal.show_cursor()?;
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
||||||
#[inline]
|
|
||||||
fn get_symbol(obj: Option<&ObjInfo>, sym: Option<SymbolRef>) -> Option<&ObjSymbol> {
|
|
||||||
Some(obj?.section_symbol(sym?).1)
|
|
||||||
}
|
|
||||||
|
|
||||||
#[inline]
|
|
||||||
fn get_symbol_diff(obj: Option<&ObjDiff>, sym: Option<SymbolRef>) -> Option<&ObjSymbolDiff> {
|
|
||||||
Some(obj?.symbol_diff(sym?))
|
|
||||||
}
|
|
||||||
|
|
||||||
fn find_function(obj: &ObjInfo, name: &str) -> Option<SymbolRef> {
|
|
||||||
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
|
|
||||||
}
|
|
||||||
|
|
||||||
#[expect(dead_code)]
|
|
||||||
struct FunctionDiffUi {
|
|
||||||
relax_reloc_diffs: bool,
|
|
||||||
left_highlight: HighlightKind,
|
|
||||||
right_highlight: HighlightKind,
|
|
||||||
scroll_x: usize,
|
|
||||||
scroll_state_x: ScrollbarState,
|
|
||||||
scroll_y: usize,
|
|
||||||
scroll_state_y: ScrollbarState,
|
|
||||||
per_page: usize,
|
|
||||||
num_rows: usize,
|
|
||||||
symbol_name: String,
|
|
||||||
target_path: Option<PathBuf>,
|
|
||||||
base_path: Option<PathBuf>,
|
|
||||||
project_config: Option<ProjectConfig>,
|
|
||||||
left_obj: Option<ObjInfo>,
|
|
||||||
right_obj: Option<ObjInfo>,
|
|
||||||
prev_obj: Option<ObjInfo>,
|
|
||||||
diff_result: DiffObjsResult,
|
|
||||||
left_sym: Option<SymbolRef>,
|
|
||||||
right_sym: Option<SymbolRef>,
|
|
||||||
prev_sym: Option<SymbolRef>,
|
|
||||||
reload_time: Option<time::OffsetDateTime>,
|
|
||||||
time_format: Vec<time::format_description::FormatItem<'static>>,
|
|
||||||
open_options: bool,
|
|
||||||
three_way: bool,
|
|
||||||
}
|
|
||||||
|
|
||||||
#[derive(Default)]
|
|
||||||
struct EventResult {
|
|
||||||
redraw: bool,
|
|
||||||
click_xy: Option<(u16, u16)>,
|
|
||||||
}
|
|
||||||
|
|
||||||
enum EventControlFlow {
|
|
||||||
Break,
|
|
||||||
Continue(EventResult),
|
|
||||||
Reload,
|
|
||||||
}
|
|
||||||
|
|
||||||
impl FunctionDiffUi {
|
|
||||||
fn draw(&mut self, 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_diff(self.diff_result.right.as_ref(), self.right_sym)
|
|
||||||
.and_then(|s| s.match_percent)
|
|
||||||
{
|
|
||||||
line_r.spans.push(Span::styled(
|
|
||||||
format!("{:.2}% ", percent),
|
|
||||||
Style::new().fg(match_percent_color(percent)),
|
|
||||||
));
|
|
||||||
}
|
|
||||||
let reload_time = self
|
|
||||||
.reload_time
|
|
||||||
.as_ref()
|
|
||||||
.and_then(|t| t.format(&self.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),
|
|
||||||
));
|
|
||||||
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), Some(symbol_diff)) = (
|
|
||||||
get_symbol(self.left_obj.as_ref(), self.left_sym),
|
|
||||||
get_symbol_diff(self.diff_result.left.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), Some(symbol_diff)) = (
|
|
||||||
get_symbol(self.right_obj.as_ref(), self.right_sym),
|
|
||||||
get_symbol_diff(self.diff_result.right.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), Some(symbol_diff)) = (
|
|
||||||
get_symbol(self.prev_obj.as_ref(), self.prev_sym),
|
|
||||||
get_symbol_diff(self.diff_result.prev.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;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
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 handle_event(&mut self, 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') => {
|
|
||||||
self.relax_reloc_diffs = !self.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 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 };
|
|
||||||
}
|
|
||||||
|
|
||||||
#[expect(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<HighlightKind> {
|
|
||||||
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(" "));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
fn reload(&mut self) -> Result<()> {
|
|
||||||
let prev = self.right_obj.take();
|
|
||||||
let config = diff::DiffObjConfig {
|
|
||||||
relax_reloc_diffs: self.relax_reloc_diffs,
|
|
||||||
..Default::default() // TODO
|
|
||||||
};
|
|
||||||
let target = self
|
|
||||||
.target_path
|
|
||||||
.as_deref()
|
|
||||||
.map(|p| {
|
|
||||||
obj::read::read(p, &config).with_context(|| format!("Loading {}", p.display()))
|
|
||||||
})
|
|
||||||
.transpose()?;
|
|
||||||
let base = self
|
|
||||||
.base_path
|
|
||||||
.as_deref()
|
|
||||||
.map(|p| {
|
|
||||||
obj::read::read(p, &config).with_context(|| format!("Loading {}", p.display()))
|
|
||||||
})
|
|
||||||
.transpose()?;
|
|
||||||
let result = diff::diff_objs(&config, target.as_ref(), base.as_ref(), prev.as_ref())?;
|
|
||||||
|
|
||||||
let left_sym = target.as_ref().and_then(|o| find_function(o, &self.symbol_name));
|
|
||||||
let right_sym = base.as_ref().and_then(|o| find_function(o, &self.symbol_name));
|
|
||||||
let prev_sym = prev.as_ref().and_then(|o| find_function(o, &self.symbol_name));
|
|
||||||
self.num_rows = match (
|
|
||||||
get_symbol_diff(result.left.as_ref(), left_sym),
|
|
||||||
get_symbol_diff(result.right.as_ref(), right_sym),
|
|
||||||
) {
|
|
||||||
(Some(l), Some(r)) => l.instructions.len().max(r.instructions.len()),
|
|
||||||
(Some(l), None) => l.instructions.len(),
|
|
||||||
(None, Some(r)) => r.instructions.len(),
|
|
||||||
(None, None) => bail!("Symbol not found: {}", self.symbol_name),
|
|
||||||
};
|
|
||||||
self.left_obj = target;
|
|
||||||
self.right_obj = base;
|
|
||||||
self.prev_obj = prev;
|
|
||||||
self.diff_result = result;
|
|
||||||
self.left_sym = left_sym;
|
|
||||||
self.right_sym = right_sym;
|
|
||||||
self.prev_sym = prev_sym;
|
|
||||||
self.reload_time = time::OffsetDateTime::now_local().ok();
|
|
||||||
Ok(())
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
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
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
|
@ -1,6 +1,7 @@
|
||||||
mod argp_version;
|
mod argp_version;
|
||||||
mod cmd;
|
mod cmd;
|
||||||
mod util;
|
mod util;
|
||||||
|
mod views;
|
||||||
|
|
||||||
// musl's allocator is very slow, so use mimalloc when targeting musl.
|
// musl's allocator is very slow, so use mimalloc when targeting musl.
|
||||||
// Otherwise, use the system allocator to avoid extra code size.
|
// Otherwise, use the system allocator to avoid extra code size.
|
||||||
|
|
|
@ -0,0 +1,652 @@
|
||||||
|
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<SymbolRef>,
|
||||||
|
pub right_sym: Option<SymbolRef>,
|
||||||
|
pub prev_sym: Option<SymbolRef>,
|
||||||
|
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.relax_reloc_diffs = !state.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<HighlightKind> {
|
||||||
|
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<SymbolRef>,
|
||||||
|
) -> 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<SymbolRef> {
|
||||||
|
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
|
||||||
|
}
|
|
@ -0,0 +1,25 @@
|
||||||
|
use anyhow::Result;
|
||||||
|
use crossterm::event::Event;
|
||||||
|
use ratatui::Frame;
|
||||||
|
|
||||||
|
use crate::cmd::diff::AppState;
|
||||||
|
|
||||||
|
pub mod function_diff;
|
||||||
|
|
||||||
|
#[derive(Default)]
|
||||||
|
pub struct EventResult {
|
||||||
|
pub redraw: bool,
|
||||||
|
pub click_xy: Option<(u16, u16)>,
|
||||||
|
}
|
||||||
|
|
||||||
|
pub enum EventControlFlow {
|
||||||
|
Break,
|
||||||
|
Continue(EventResult),
|
||||||
|
Reload,
|
||||||
|
}
|
||||||
|
|
||||||
|
pub trait UiView {
|
||||||
|
fn draw(&mut self, state: &AppState, f: &mut Frame, result: &mut EventResult);
|
||||||
|
fn handle_event(&mut self, state: &mut AppState, event: Event) -> EventControlFlow;
|
||||||
|
fn reload(&mut self, state: &AppState) -> Result<()>;
|
||||||
|
}
|
|
@ -16,8 +16,10 @@ documentation = "https://docs.rs/objdiff-core"
|
||||||
crate-type = ["cdylib", "rlib"]
|
crate-type = ["cdylib", "rlib"]
|
||||||
|
|
||||||
[features]
|
[features]
|
||||||
all = ["config", "dwarf", "mips", "ppc", "x86", "arm", "arm64", "bindings"]
|
all = ["config", "dwarf", "mips", "ppc", "x86", "arm", "arm64", "bindings", "build"]
|
||||||
any-arch = ["config", "dep:bimap", "dep:strum", "dep:similar", "dep:flagset", "dep:log", "dep:memmap2", "dep:byteorder", "dep:num-traits"] # Implicit, used to check if any arch is enabled
|
any-arch = ["config", "dep:bimap", "dep:strum", "dep:similar", "dep:flagset", "dep:log", "dep:memmap2", "dep:byteorder", "dep:num-traits"] # Implicit, used to check if any arch is enabled
|
||||||
|
bindings = ["dep:serde_json", "dep:prost", "dep:pbjson", "dep:serde", "dep:prost-build", "dep:pbjson-build"]
|
||||||
|
build = ["dep:shell-escape", "dep:path-slash", "dep:winapi", "dep:notify", "dep:notify-debouncer-full", "dep:reqwest", "dep:self_update", "dep:tempfile", "dep:time"]
|
||||||
config = ["dep:bimap", "dep:globset", "dep:semver", "dep:serde_json", "dep:serde_yaml", "dep:serde", "dep:filetime"]
|
config = ["dep:bimap", "dep:globset", "dep:semver", "dep:serde_json", "dep:serde_yaml", "dep:serde", "dep:filetime"]
|
||||||
dwarf = ["dep:gimli"]
|
dwarf = ["dep:gimli"]
|
||||||
mips = ["any-arch", "dep:rabbitizer"]
|
mips = ["any-arch", "dep:rabbitizer"]
|
||||||
|
@ -25,7 +27,6 @@ ppc = ["any-arch", "dep:cwdemangle", "dep:cwextab", "dep:ppc750cl"]
|
||||||
x86 = ["any-arch", "dep:cpp_demangle", "dep:iced-x86", "dep:msvc-demangler"]
|
x86 = ["any-arch", "dep:cpp_demangle", "dep:iced-x86", "dep:msvc-demangler"]
|
||||||
arm = ["any-arch", "dep:cpp_demangle", "dep:unarm", "dep:arm-attr"]
|
arm = ["any-arch", "dep:cpp_demangle", "dep:unarm", "dep:arm-attr"]
|
||||||
arm64 = ["any-arch", "dep:cpp_demangle", "dep:yaxpeax-arch", "dep:yaxpeax-arm"]
|
arm64 = ["any-arch", "dep:cpp_demangle", "dep:yaxpeax-arch", "dep:yaxpeax-arm"]
|
||||||
bindings = ["dep:serde_json", "dep:prost", "dep:pbjson", "dep:serde", "dep:prost-build", "dep:pbjson-build"]
|
|
||||||
wasm = ["bindings", "any-arch", "dep:console_error_panic_hook", "dep:console_log", "dep:wasm-bindgen", "dep:tsify-next", "dep:log"]
|
wasm = ["bindings", "any-arch", "dep:console_error_panic_hook", "dep:console_log", "dep:wasm-bindgen", "dep:tsify-next", "dep:log"]
|
||||||
|
|
||||||
[package.metadata.docs.rs]
|
[package.metadata.docs.rs]
|
||||||
|
@ -81,6 +82,27 @@ arm-attr = { version = "0.1", optional = true }
|
||||||
yaxpeax-arch = { version = "0.3", default-features = false, features = ["std"], optional = true }
|
yaxpeax-arch = { version = "0.3", default-features = false, features = ["std"], optional = true }
|
||||||
yaxpeax-arm = { version = "0.3", default-features = false, features = ["std"], optional = true }
|
yaxpeax-arm = { version = "0.3", default-features = false, features = ["std"], optional = true }
|
||||||
|
|
||||||
|
# build
|
||||||
|
notify = { git = "https://github.com/notify-rs/notify", rev = "128bf6230c03d39dbb7f301ff7b20e594e34c3a2", optional = true }
|
||||||
|
notify-debouncer-full = { git = "https://github.com/notify-rs/notify", rev = "128bf6230c03d39dbb7f301ff7b20e594e34c3a2", optional = true }
|
||||||
|
shell-escape = { version = "0.1", optional = true }
|
||||||
|
tempfile = { version = "3.13", optional = true }
|
||||||
|
time = { version = "0.3", optional = true }
|
||||||
|
|
||||||
|
[target.'cfg(windows)'.dependencies]
|
||||||
|
path-slash = { version = "0.2", optional = true }
|
||||||
|
winapi = { version = "0.3", optional = true }
|
||||||
|
|
||||||
|
# For Linux static binaries, use rustls
|
||||||
|
[target.'cfg(target_os = "linux")'.dependencies]
|
||||||
|
reqwest = { version = "0.12", default-features = false, features = ["blocking", "json", "multipart", "rustls-tls"], optional = true }
|
||||||
|
self_update = { version = "0.41", default-features = false, features = ["rustls"], optional = true }
|
||||||
|
|
||||||
|
# For all other platforms, use native TLS
|
||||||
|
[target.'cfg(not(target_os = "linux"))'.dependencies]
|
||||||
|
reqwest = { version = "0.12", default-features = false, features = ["blocking", "json", "multipart", "default-tls"], optional = true }
|
||||||
|
self_update = { version = "0.41", optional = true }
|
||||||
|
|
||||||
[build-dependencies]
|
[build-dependencies]
|
||||||
prost-build = { version = "0.13", optional = true }
|
prost-build = { version = "0.13", optional = true }
|
||||||
pbjson-build = { version = "0.7", optional = true }
|
pbjson-build = { version = "0.7", optional = true }
|
||||||
|
|
|
@ -0,0 +1,106 @@
|
||||||
|
pub mod watcher;
|
||||||
|
|
||||||
|
use std::{
|
||||||
|
path::{Path, PathBuf},
|
||||||
|
process::Command,
|
||||||
|
};
|
||||||
|
|
||||||
|
pub struct BuildStatus {
|
||||||
|
pub success: bool,
|
||||||
|
pub cmdline: String,
|
||||||
|
pub stdout: String,
|
||||||
|
pub stderr: String,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Default for BuildStatus {
|
||||||
|
fn default() -> Self {
|
||||||
|
BuildStatus {
|
||||||
|
success: true,
|
||||||
|
cmdline: String::new(),
|
||||||
|
stdout: String::new(),
|
||||||
|
stderr: String::new(),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(Debug, Clone)]
|
||||||
|
pub struct BuildConfig {
|
||||||
|
pub project_dir: Option<PathBuf>,
|
||||||
|
pub custom_make: Option<String>,
|
||||||
|
pub custom_args: Option<Vec<String>>,
|
||||||
|
#[allow(unused)]
|
||||||
|
pub selected_wsl_distro: Option<String>,
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn run_make(config: &BuildConfig, arg: &Path) -> BuildStatus {
|
||||||
|
let Some(cwd) = &config.project_dir else {
|
||||||
|
return BuildStatus {
|
||||||
|
success: false,
|
||||||
|
stderr: "Missing project dir".to_string(),
|
||||||
|
..Default::default()
|
||||||
|
};
|
||||||
|
};
|
||||||
|
let make = config.custom_make.as_deref().unwrap_or("make");
|
||||||
|
let make_args = config.custom_args.as_deref().unwrap_or(&[]);
|
||||||
|
#[cfg(not(windows))]
|
||||||
|
let mut command = {
|
||||||
|
let mut command = Command::new(make);
|
||||||
|
command.current_dir(cwd).args(make_args).arg(arg);
|
||||||
|
command
|
||||||
|
};
|
||||||
|
#[cfg(windows)]
|
||||||
|
let mut command = {
|
||||||
|
use std::os::windows::process::CommandExt;
|
||||||
|
|
||||||
|
use path_slash::PathExt;
|
||||||
|
let mut command = if config.selected_wsl_distro.is_some() {
|
||||||
|
Command::new("wsl")
|
||||||
|
} else {
|
||||||
|
Command::new(make)
|
||||||
|
};
|
||||||
|
if let Some(distro) = &config.selected_wsl_distro {
|
||||||
|
// Strip distro root prefix \\wsl.localhost\{distro}
|
||||||
|
let wsl_path_prefix = format!("\\\\wsl.localhost\\{}", distro);
|
||||||
|
let cwd = match cwd.strip_prefix(wsl_path_prefix) {
|
||||||
|
Ok(new_cwd) => format!("/{}", new_cwd.to_slash_lossy().as_ref()),
|
||||||
|
Err(_) => cwd.to_string_lossy().to_string(),
|
||||||
|
};
|
||||||
|
|
||||||
|
command
|
||||||
|
.arg("--cd")
|
||||||
|
.arg(cwd)
|
||||||
|
.arg("-d")
|
||||||
|
.arg(distro)
|
||||||
|
.arg("--")
|
||||||
|
.arg(make)
|
||||||
|
.args(make_args)
|
||||||
|
.arg(arg.to_slash_lossy().as_ref());
|
||||||
|
} else {
|
||||||
|
command.current_dir(cwd).args(make_args).arg(arg.to_slash_lossy().as_ref());
|
||||||
|
}
|
||||||
|
command.creation_flags(winapi::um::winbase::CREATE_NO_WINDOW);
|
||||||
|
command
|
||||||
|
};
|
||||||
|
let mut cmdline = shell_escape::escape(command.get_program().to_string_lossy()).into_owned();
|
||||||
|
for arg in command.get_args() {
|
||||||
|
cmdline.push(' ');
|
||||||
|
cmdline.push_str(shell_escape::escape(arg.to_string_lossy()).as_ref());
|
||||||
|
}
|
||||||
|
let output = match command.output() {
|
||||||
|
Ok(output) => output,
|
||||||
|
Err(e) => {
|
||||||
|
return BuildStatus {
|
||||||
|
success: false,
|
||||||
|
cmdline,
|
||||||
|
stdout: Default::default(),
|
||||||
|
stderr: e.to_string(),
|
||||||
|
};
|
||||||
|
}
|
||||||
|
};
|
||||||
|
// Try from_utf8 first to avoid copying the buffer if it's valid, then fall back to from_utf8_lossy
|
||||||
|
let stdout = String::from_utf8(output.stdout)
|
||||||
|
.unwrap_or_else(|e| String::from_utf8_lossy(e.as_bytes()).into_owned());
|
||||||
|
let stderr = String::from_utf8(output.stderr)
|
||||||
|
.unwrap_or_else(|e| String::from_utf8_lossy(e.as_bytes()).into_owned());
|
||||||
|
BuildStatus { success: output.status.success(), cmdline, stdout, stderr }
|
||||||
|
}
|
|
@ -0,0 +1,75 @@
|
||||||
|
use std::{
|
||||||
|
fs,
|
||||||
|
path::{Path, PathBuf},
|
||||||
|
sync::{
|
||||||
|
atomic::{AtomicBool, Ordering},
|
||||||
|
Arc,
|
||||||
|
},
|
||||||
|
task::Waker,
|
||||||
|
time::Duration,
|
||||||
|
};
|
||||||
|
|
||||||
|
use globset::GlobSet;
|
||||||
|
use notify::RecursiveMode;
|
||||||
|
use notify_debouncer_full::{new_debouncer_opt, DebounceEventResult};
|
||||||
|
|
||||||
|
pub type Watcher = notify_debouncer_full::Debouncer<
|
||||||
|
notify::RecommendedWatcher,
|
||||||
|
notify_debouncer_full::RecommendedCache,
|
||||||
|
>;
|
||||||
|
|
||||||
|
pub struct WatcherState {
|
||||||
|
pub config_path: Option<PathBuf>,
|
||||||
|
pub left_obj_path: Option<PathBuf>,
|
||||||
|
pub right_obj_path: Option<PathBuf>,
|
||||||
|
pub patterns: GlobSet,
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn create_watcher(
|
||||||
|
modified: Arc<AtomicBool>,
|
||||||
|
project_dir: &Path,
|
||||||
|
patterns: GlobSet,
|
||||||
|
waker: Waker,
|
||||||
|
) -> notify::Result<Watcher> {
|
||||||
|
let base_dir = fs::canonicalize(project_dir)?;
|
||||||
|
let base_dir_clone = base_dir.clone();
|
||||||
|
let timeout = Duration::from_millis(200);
|
||||||
|
let config = notify::Config::default().with_poll_interval(Duration::from_secs(2));
|
||||||
|
let mut debouncer = new_debouncer_opt(
|
||||||
|
timeout,
|
||||||
|
None,
|
||||||
|
move |result: DebounceEventResult| match result {
|
||||||
|
Ok(events) => {
|
||||||
|
let mut any_match = false;
|
||||||
|
for event in events.iter() {
|
||||||
|
if !matches!(
|
||||||
|
event.kind,
|
||||||
|
notify::EventKind::Modify(..)
|
||||||
|
| notify::EventKind::Create(..)
|
||||||
|
| notify::EventKind::Remove(..)
|
||||||
|
) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
for path in &event.paths {
|
||||||
|
let Ok(path) = path.strip_prefix(&base_dir_clone) else {
|
||||||
|
continue;
|
||||||
|
};
|
||||||
|
if patterns.is_match(path) {
|
||||||
|
// log::info!("File modified: {}", path.display());
|
||||||
|
any_match = true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if any_match {
|
||||||
|
modified.store(true, Ordering::Relaxed);
|
||||||
|
waker.wake_by_ref();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
Err(errors) => errors.iter().for_each(|e| log::error!("Watch error: {e:?}")),
|
||||||
|
},
|
||||||
|
notify_debouncer_full::RecommendedCache::new(),
|
||||||
|
config,
|
||||||
|
)?;
|
||||||
|
debouncer.watch(base_dir, RecursiveMode::Recursive)?;
|
||||||
|
Ok(debouncer)
|
||||||
|
}
|
|
@ -176,6 +176,10 @@ pub const DEFAULT_WATCH_PATTERNS: &[&str] = &[
|
||||||
"*.inc", "*.py", "*.yml", "*.txt", "*.json",
|
"*.inc", "*.py", "*.yml", "*.txt", "*.json",
|
||||||
];
|
];
|
||||||
|
|
||||||
|
pub fn default_watch_patterns() -> Vec<Glob> {
|
||||||
|
DEFAULT_WATCH_PATTERNS.iter().map(|s| Glob::new(s).unwrap()).collect()
|
||||||
|
}
|
||||||
|
|
||||||
#[derive(Clone, Eq, PartialEq)]
|
#[derive(Clone, Eq, PartialEq)]
|
||||||
pub struct ProjectConfigInfo {
|
pub struct ProjectConfigInfo {
|
||||||
pub path: PathBuf,
|
pub path: PathBuf,
|
||||||
|
|
|
@ -0,0 +1,50 @@
|
||||||
|
use std::{sync::mpsc::Receiver, task::Waker};
|
||||||
|
|
||||||
|
use anyhow::{Context, Result};
|
||||||
|
use self_update::{
|
||||||
|
cargo_crate_version,
|
||||||
|
update::{Release, ReleaseUpdate},
|
||||||
|
};
|
||||||
|
|
||||||
|
use crate::jobs::{start_job, update_status, Job, JobContext, JobResult, JobState};
|
||||||
|
|
||||||
|
pub struct CheckUpdateConfig {
|
||||||
|
pub build_updater: fn() -> Result<Box<dyn ReleaseUpdate>>,
|
||||||
|
pub bin_names: Vec<String>,
|
||||||
|
}
|
||||||
|
|
||||||
|
pub struct CheckUpdateResult {
|
||||||
|
pub update_available: bool,
|
||||||
|
pub latest_release: Release,
|
||||||
|
pub found_binary: Option<String>,
|
||||||
|
}
|
||||||
|
|
||||||
|
fn run_check_update(
|
||||||
|
context: &JobContext,
|
||||||
|
cancel: Receiver<()>,
|
||||||
|
config: CheckUpdateConfig,
|
||||||
|
) -> Result<Box<CheckUpdateResult>> {
|
||||||
|
update_status(context, "Fetching latest release".to_string(), 0, 1, &cancel)?;
|
||||||
|
let updater = (config.build_updater)().context("Failed to create release updater")?;
|
||||||
|
let latest_release = updater.get_latest_release()?;
|
||||||
|
let update_available =
|
||||||
|
self_update::version::bump_is_greater(cargo_crate_version!(), &latest_release.version)?;
|
||||||
|
// Find the binary name in the release assets
|
||||||
|
let mut found_binary = None;
|
||||||
|
for bin_name in &config.bin_names {
|
||||||
|
if latest_release.assets.iter().any(|a| &a.name == bin_name) {
|
||||||
|
found_binary = Some(bin_name.clone());
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
update_status(context, "Complete".to_string(), 1, 1, &cancel)?;
|
||||||
|
Ok(Box::new(CheckUpdateResult { update_available, latest_release, found_binary }))
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn start_check_update(waker: Waker, config: CheckUpdateConfig) -> JobState {
|
||||||
|
start_job(waker, "Check for updates", Job::CheckUpdate, move |context, cancel| {
|
||||||
|
run_check_update(&context, cancel, config)
|
||||||
|
.map(|result| JobResult::CheckUpdate(Some(result)))
|
||||||
|
})
|
||||||
|
}
|
|
@ -1,14 +1,10 @@
|
||||||
use std::{fs, path::PathBuf, sync::mpsc::Receiver};
|
use std::{fs, path::PathBuf, sync::mpsc::Receiver, task::Waker};
|
||||||
|
|
||||||
use anyhow::{anyhow, bail, Context, Result};
|
use anyhow::{anyhow, bail, Context, Result};
|
||||||
use const_format::formatcp;
|
|
||||||
|
|
||||||
use crate::{
|
use crate::{
|
||||||
app::AppConfig,
|
build::{run_make, BuildConfig, BuildStatus},
|
||||||
jobs::{
|
jobs::{start_job, update_status, Job, JobContext, JobResult, JobState},
|
||||||
objdiff::{run_make, BuildConfig, BuildStatus},
|
|
||||||
start_job, update_status, Job, JobContext, JobResult, JobState,
|
|
||||||
},
|
|
||||||
};
|
};
|
||||||
|
|
||||||
#[derive(Debug, Clone)]
|
#[derive(Debug, Clone)]
|
||||||
|
@ -26,38 +22,6 @@ pub struct CreateScratchConfig {
|
||||||
pub preset_id: Option<u32>,
|
pub preset_id: Option<u32>,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl CreateScratchConfig {
|
|
||||||
pub(crate) fn from_config(config: &AppConfig, function_name: String) -> Result<Self> {
|
|
||||||
let Some(selected_obj) = &config.selected_obj else {
|
|
||||||
bail!("No object selected");
|
|
||||||
};
|
|
||||||
let Some(target_path) = &selected_obj.target_path else {
|
|
||||||
bail!("No target path for {}", selected_obj.name);
|
|
||||||
};
|
|
||||||
let Some(scratch_config) = &selected_obj.scratch else {
|
|
||||||
bail!("No scratch configuration for {}", selected_obj.name);
|
|
||||||
};
|
|
||||||
Ok(Self {
|
|
||||||
build_config: BuildConfig::from_config(config),
|
|
||||||
context_path: scratch_config.ctx_path.clone(),
|
|
||||||
build_context: scratch_config.build_ctx.unwrap_or(false),
|
|
||||||
compiler: scratch_config.compiler.clone().unwrap_or_default(),
|
|
||||||
platform: scratch_config.platform.clone().unwrap_or_default(),
|
|
||||||
compiler_flags: scratch_config.c_flags.clone().unwrap_or_default(),
|
|
||||||
function_name,
|
|
||||||
target_obj: target_path.to_path_buf(),
|
|
||||||
preset_id: scratch_config.preset_id,
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn is_available(config: &AppConfig) -> bool {
|
|
||||||
let Some(selected_obj) = &config.selected_obj else {
|
|
||||||
return false;
|
|
||||||
};
|
|
||||||
selected_obj.target_path.is_some() && selected_obj.scratch.is_some()
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
#[derive(Default, Debug, Clone)]
|
#[derive(Default, Debug, Clone)]
|
||||||
pub struct CreateScratchResult {
|
pub struct CreateScratchResult {
|
||||||
pub scratch_url: String,
|
pub scratch_url: String,
|
||||||
|
@ -99,7 +63,7 @@ fn run_create_scratch(
|
||||||
|
|
||||||
update_status(status, "Creating scratch".to_string(), 1, 2, &cancel)?;
|
update_status(status, "Creating scratch".to_string(), 1, 2, &cancel)?;
|
||||||
let diff_flags = [format!("--disassemble={}", config.function_name)];
|
let diff_flags = [format!("--disassemble={}", config.function_name)];
|
||||||
let diff_flags = serde_json::to_string(&diff_flags).unwrap();
|
let diff_flags = serde_json::to_string(&diff_flags)?;
|
||||||
let obj_path = project_dir.join(&config.target_obj);
|
let obj_path = project_dir.join(&config.target_obj);
|
||||||
let file = reqwest::blocking::multipart::Part::file(&obj_path)
|
let file = reqwest::blocking::multipart::Part::file(&obj_path)
|
||||||
.with_context(|| format!("Failed to open {}", obj_path.display()))?;
|
.with_context(|| format!("Failed to open {}", obj_path.display()))?;
|
||||||
|
@ -117,7 +81,7 @@ fn run_create_scratch(
|
||||||
form = form.part("target_obj", file);
|
form = form.part("target_obj", file);
|
||||||
let client = reqwest::blocking::Client::new();
|
let client = reqwest::blocking::Client::new();
|
||||||
let response = client
|
let response = client
|
||||||
.post(formatcp!("{API_HOST}/api/scratch"))
|
.post(format!("{API_HOST}/api/scratch"))
|
||||||
.multipart(form)
|
.multipart(form)
|
||||||
.send()
|
.send()
|
||||||
.map_err(|e| anyhow!("Failed to send request: {}", e))?;
|
.map_err(|e| anyhow!("Failed to send request: {}", e))?;
|
||||||
|
@ -131,8 +95,8 @@ fn run_create_scratch(
|
||||||
Ok(Box::from(CreateScratchResult { scratch_url }))
|
Ok(Box::from(CreateScratchResult { scratch_url }))
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn start_create_scratch(ctx: &egui::Context, config: CreateScratchConfig) -> JobState {
|
pub fn start_create_scratch(waker: Waker, config: CreateScratchConfig) -> JobState {
|
||||||
start_job(ctx, "Create scratch", Job::CreateScratch, move |context, cancel| {
|
start_job(waker, "Create scratch", Job::CreateScratch, move |context, cancel| {
|
||||||
run_create_scratch(&context, cancel, config)
|
run_create_scratch(&context, cancel, config)
|
||||||
.map(|result| JobResult::CreateScratch(Some(result)))
|
.map(|result| JobResult::CreateScratch(Some(result)))
|
||||||
})
|
})
|
|
@ -4,6 +4,7 @@ use std::{
|
||||||
mpsc::{Receiver, Sender, TryRecvError},
|
mpsc::{Receiver, Sender, TryRecvError},
|
||||||
Arc, RwLock,
|
Arc, RwLock,
|
||||||
},
|
},
|
||||||
|
task::Waker,
|
||||||
thread::JoinHandle,
|
thread::JoinHandle,
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -53,7 +54,6 @@ impl JobQueue {
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Returns whether any job is running.
|
/// Returns whether any job is running.
|
||||||
#[expect(dead_code)]
|
|
||||||
pub fn any_running(&self) -> bool {
|
pub fn any_running(&self) -> bool {
|
||||||
self.jobs.iter().any(|job| {
|
self.jobs.iter().any(|job| {
|
||||||
if let Some(handle) = &job.handle {
|
if let Some(handle) = &job.handle {
|
||||||
|
@ -96,12 +96,53 @@ impl JobQueue {
|
||||||
|
|
||||||
/// Removes a job from the queue given its ID.
|
/// Removes a job from the queue given its ID.
|
||||||
pub fn remove(&mut self, id: usize) { self.jobs.retain(|job| job.id != id); }
|
pub fn remove(&mut self, id: usize) { self.jobs.retain(|job| job.id != id); }
|
||||||
|
|
||||||
|
/// Collects the results of all finished jobs and handles any errors.
|
||||||
|
pub fn collect_results(&mut self) {
|
||||||
|
let mut results = vec![];
|
||||||
|
for (job, result) in self.iter_finished() {
|
||||||
|
match result {
|
||||||
|
Ok(result) => {
|
||||||
|
match result {
|
||||||
|
JobResult::None => {
|
||||||
|
// Job context contains the error
|
||||||
|
}
|
||||||
|
_ => results.push(result),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
Err(err) => {
|
||||||
|
let err = if let Some(msg) = err.downcast_ref::<&'static str>() {
|
||||||
|
anyhow::Error::msg(*msg)
|
||||||
|
} else if let Some(msg) = err.downcast_ref::<String>() {
|
||||||
|
anyhow::Error::msg(msg.clone())
|
||||||
|
} else {
|
||||||
|
anyhow::Error::msg("Thread panicked")
|
||||||
|
};
|
||||||
|
let result = job.context.status.write();
|
||||||
|
if let Ok(mut guard) = result {
|
||||||
|
guard.error = Some(err);
|
||||||
|
} else {
|
||||||
|
drop(result);
|
||||||
|
job.context.status = Arc::new(RwLock::new(JobStatus {
|
||||||
|
title: "Error".to_string(),
|
||||||
|
progress_percent: 0.0,
|
||||||
|
progress_items: None,
|
||||||
|
status: String::new(),
|
||||||
|
error: Some(err),
|
||||||
|
}));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
self.results.append(&mut results);
|
||||||
|
self.clear_finished();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Clone)]
|
#[derive(Clone)]
|
||||||
pub struct JobContext {
|
pub struct JobContext {
|
||||||
pub status: Arc<RwLock<JobStatus>>,
|
pub status: Arc<RwLock<JobStatus>>,
|
||||||
pub egui: egui::Context,
|
pub waker: Waker,
|
||||||
}
|
}
|
||||||
|
|
||||||
pub struct JobState {
|
pub struct JobState {
|
||||||
|
@ -137,7 +178,7 @@ fn should_cancel(rx: &Receiver<()>) -> bool {
|
||||||
}
|
}
|
||||||
|
|
||||||
fn start_job(
|
fn start_job(
|
||||||
ctx: &egui::Context,
|
waker: Waker,
|
||||||
title: &str,
|
title: &str,
|
||||||
kind: Job,
|
kind: Job,
|
||||||
run: impl FnOnce(JobContext, Receiver<()>) -> Result<JobResult> + Send + 'static,
|
run: impl FnOnce(JobContext, Receiver<()>) -> Result<JobResult> + Send + 'static,
|
||||||
|
@ -149,8 +190,8 @@ fn start_job(
|
||||||
status: String::new(),
|
status: String::new(),
|
||||||
error: None,
|
error: None,
|
||||||
}));
|
}));
|
||||||
let context = JobContext { status: status.clone(), egui: ctx.clone() };
|
let context = JobContext { status: status.clone(), waker: waker.clone() };
|
||||||
let context_inner = JobContext { status: status.clone(), egui: ctx.clone() };
|
let context_inner = JobContext { status: status.clone(), waker };
|
||||||
let (tx, rx) = std::sync::mpsc::channel();
|
let (tx, rx) = std::sync::mpsc::channel();
|
||||||
let handle = std::thread::spawn(move || match run(context_inner, rx) {
|
let handle = std::thread::spawn(move || match run(context_inner, rx) {
|
||||||
Ok(state) => state,
|
Ok(state) => state,
|
||||||
|
@ -162,7 +203,7 @@ fn start_job(
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
let id = JOB_ID.fetch_add(1, Ordering::Relaxed);
|
let id = JOB_ID.fetch_add(1, Ordering::Relaxed);
|
||||||
log::info!("Started job {}", id);
|
// log::info!("Started job {}", id); TODO
|
||||||
JobState { id, kind, handle: Some(handle), context, cancel: tx }
|
JobState { id, kind, handle: Some(handle), context, cancel: tx }
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -184,6 +225,6 @@ fn update_status(
|
||||||
w.status = str;
|
w.status = str;
|
||||||
}
|
}
|
||||||
drop(w);
|
drop(w);
|
||||||
context.egui.request_repaint();
|
context.waker.wake_by_ref();
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
|
@ -0,0 +1,199 @@
|
||||||
|
use std::{path::PathBuf, sync::mpsc::Receiver, task::Waker};
|
||||||
|
|
||||||
|
use anyhow::{anyhow, Error, Result};
|
||||||
|
use time::OffsetDateTime;
|
||||||
|
|
||||||
|
use crate::{
|
||||||
|
build::{run_make, BuildConfig, BuildStatus},
|
||||||
|
config::SymbolMappings,
|
||||||
|
diff::{diff_objs, DiffObjConfig, MappingConfig, ObjDiff},
|
||||||
|
jobs::{start_job, update_status, Job, JobContext, JobResult, JobState},
|
||||||
|
obj::{read, ObjInfo},
|
||||||
|
};
|
||||||
|
|
||||||
|
pub struct ObjDiffConfig {
|
||||||
|
pub build_config: BuildConfig,
|
||||||
|
pub build_base: bool,
|
||||||
|
pub build_target: bool,
|
||||||
|
pub target_path: Option<PathBuf>,
|
||||||
|
pub base_path: Option<PathBuf>,
|
||||||
|
pub diff_obj_config: DiffObjConfig,
|
||||||
|
pub symbol_mappings: SymbolMappings,
|
||||||
|
pub selecting_left: Option<String>,
|
||||||
|
pub selecting_right: Option<String>,
|
||||||
|
}
|
||||||
|
|
||||||
|
pub struct ObjDiffResult {
|
||||||
|
pub first_status: BuildStatus,
|
||||||
|
pub second_status: BuildStatus,
|
||||||
|
pub first_obj: Option<(ObjInfo, ObjDiff)>,
|
||||||
|
pub second_obj: Option<(ObjInfo, ObjDiff)>,
|
||||||
|
pub time: OffsetDateTime,
|
||||||
|
}
|
||||||
|
|
||||||
|
fn run_build(
|
||||||
|
context: &JobContext,
|
||||||
|
cancel: Receiver<()>,
|
||||||
|
mut config: ObjDiffConfig,
|
||||||
|
) -> Result<Box<ObjDiffResult>> {
|
||||||
|
// Use the per-object symbol mappings, we don't set mappings globally
|
||||||
|
config.diff_obj_config.symbol_mappings = MappingConfig {
|
||||||
|
mappings: config.symbol_mappings,
|
||||||
|
selecting_left: config.selecting_left,
|
||||||
|
selecting_right: config.selecting_right,
|
||||||
|
};
|
||||||
|
|
||||||
|
let mut target_path_rel = None;
|
||||||
|
let mut base_path_rel = None;
|
||||||
|
if config.build_target || config.build_base {
|
||||||
|
let project_dir = config
|
||||||
|
.build_config
|
||||||
|
.project_dir
|
||||||
|
.as_ref()
|
||||||
|
.ok_or_else(|| Error::msg("Missing project dir"))?;
|
||||||
|
if let Some(target_path) = &config.target_path {
|
||||||
|
target_path_rel = Some(target_path.strip_prefix(project_dir).map_err(|_| {
|
||||||
|
anyhow!(
|
||||||
|
"Target path '{}' doesn't begin with '{}'",
|
||||||
|
target_path.display(),
|
||||||
|
project_dir.display()
|
||||||
|
)
|
||||||
|
})?);
|
||||||
|
}
|
||||||
|
if let Some(base_path) = &config.base_path {
|
||||||
|
base_path_rel = Some(base_path.strip_prefix(project_dir).map_err(|_| {
|
||||||
|
anyhow!(
|
||||||
|
"Base path '{}' doesn't begin with '{}'",
|
||||||
|
base_path.display(),
|
||||||
|
project_dir.display()
|
||||||
|
)
|
||||||
|
})?);
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
let mut total = 1;
|
||||||
|
if config.build_target && target_path_rel.is_some() {
|
||||||
|
total += 1;
|
||||||
|
}
|
||||||
|
if config.build_base && base_path_rel.is_some() {
|
||||||
|
total += 1;
|
||||||
|
}
|
||||||
|
if config.target_path.is_some() {
|
||||||
|
total += 1;
|
||||||
|
}
|
||||||
|
if config.base_path.is_some() {
|
||||||
|
total += 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
let mut step_idx = 0;
|
||||||
|
let mut first_status = match target_path_rel {
|
||||||
|
Some(target_path_rel) if config.build_target => {
|
||||||
|
update_status(
|
||||||
|
context,
|
||||||
|
format!("Building target {}", target_path_rel.display()),
|
||||||
|
step_idx,
|
||||||
|
total,
|
||||||
|
&cancel,
|
||||||
|
)?;
|
||||||
|
step_idx += 1;
|
||||||
|
run_make(&config.build_config, target_path_rel)
|
||||||
|
}
|
||||||
|
_ => BuildStatus::default(),
|
||||||
|
};
|
||||||
|
|
||||||
|
let mut second_status = match base_path_rel {
|
||||||
|
Some(base_path_rel) if config.build_base => {
|
||||||
|
update_status(
|
||||||
|
context,
|
||||||
|
format!("Building base {}", base_path_rel.display()),
|
||||||
|
step_idx,
|
||||||
|
total,
|
||||||
|
&cancel,
|
||||||
|
)?;
|
||||||
|
step_idx += 1;
|
||||||
|
run_make(&config.build_config, base_path_rel)
|
||||||
|
}
|
||||||
|
_ => BuildStatus::default(),
|
||||||
|
};
|
||||||
|
|
||||||
|
let time = OffsetDateTime::now_utc();
|
||||||
|
|
||||||
|
let first_obj = match &config.target_path {
|
||||||
|
Some(target_path) if first_status.success => {
|
||||||
|
update_status(
|
||||||
|
context,
|
||||||
|
format!("Loading target {}", target_path.display()),
|
||||||
|
step_idx,
|
||||||
|
total,
|
||||||
|
&cancel,
|
||||||
|
)?;
|
||||||
|
step_idx += 1;
|
||||||
|
match read::read(target_path, &config.diff_obj_config) {
|
||||||
|
Ok(obj) => Some(obj),
|
||||||
|
Err(e) => {
|
||||||
|
first_status = BuildStatus {
|
||||||
|
success: false,
|
||||||
|
stdout: format!("Loading object '{}'", target_path.display()),
|
||||||
|
stderr: format!("{:#}", e),
|
||||||
|
..Default::default()
|
||||||
|
};
|
||||||
|
None
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
Some(_) => {
|
||||||
|
step_idx += 1;
|
||||||
|
None
|
||||||
|
}
|
||||||
|
_ => None,
|
||||||
|
};
|
||||||
|
|
||||||
|
let second_obj = match &config.base_path {
|
||||||
|
Some(base_path) if second_status.success => {
|
||||||
|
update_status(
|
||||||
|
context,
|
||||||
|
format!("Loading base {}", base_path.display()),
|
||||||
|
step_idx,
|
||||||
|
total,
|
||||||
|
&cancel,
|
||||||
|
)?;
|
||||||
|
step_idx += 1;
|
||||||
|
match read::read(base_path, &config.diff_obj_config) {
|
||||||
|
Ok(obj) => Some(obj),
|
||||||
|
Err(e) => {
|
||||||
|
second_status = BuildStatus {
|
||||||
|
success: false,
|
||||||
|
stdout: format!("Loading object '{}'", base_path.display()),
|
||||||
|
stderr: format!("{:#}", e),
|
||||||
|
..Default::default()
|
||||||
|
};
|
||||||
|
None
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
Some(_) => {
|
||||||
|
step_idx += 1;
|
||||||
|
None
|
||||||
|
}
|
||||||
|
_ => None,
|
||||||
|
};
|
||||||
|
|
||||||
|
update_status(context, "Performing diff".to_string(), step_idx, total, &cancel)?;
|
||||||
|
step_idx += 1;
|
||||||
|
let result = diff_objs(&config.diff_obj_config, first_obj.as_ref(), second_obj.as_ref(), None)?;
|
||||||
|
|
||||||
|
update_status(context, "Complete".to_string(), step_idx, total, &cancel)?;
|
||||||
|
Ok(Box::new(ObjDiffResult {
|
||||||
|
first_status,
|
||||||
|
second_status,
|
||||||
|
first_obj: first_obj.and_then(|o| result.left.map(|d| (o, d))),
|
||||||
|
second_obj: second_obj.and_then(|o| result.right.map(|d| (o, d))),
|
||||||
|
time,
|
||||||
|
}))
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn start_build(waker: Waker, config: ObjDiffConfig) -> JobState {
|
||||||
|
start_job(waker, "Build", Job::ObjDiff, move |context, cancel| {
|
||||||
|
run_build(&context, cancel, config).map(|result| JobResult::ObjDiff(Some(result)))
|
||||||
|
})
|
||||||
|
}
|
|
@ -3,14 +3,19 @@ use std::{
|
||||||
fs::File,
|
fs::File,
|
||||||
path::PathBuf,
|
path::PathBuf,
|
||||||
sync::mpsc::Receiver,
|
sync::mpsc::Receiver,
|
||||||
|
task::Waker,
|
||||||
};
|
};
|
||||||
|
|
||||||
use anyhow::{Context, Result};
|
use anyhow::{Context, Result};
|
||||||
|
pub use self_update; // Re-export self_update crate
|
||||||
|
use self_update::update::ReleaseUpdate;
|
||||||
|
|
||||||
use crate::{
|
use crate::jobs::{start_job, update_status, Job, JobContext, JobResult, JobState};
|
||||||
jobs::{start_job, update_status, Job, JobContext, JobResult, JobState},
|
|
||||||
update::build_updater,
|
pub struct UpdateConfig {
|
||||||
};
|
pub build_updater: fn() -> Result<Box<dyn ReleaseUpdate>>,
|
||||||
|
pub bin_name: String,
|
||||||
|
}
|
||||||
|
|
||||||
pub struct UpdateResult {
|
pub struct UpdateResult {
|
||||||
pub exe_path: PathBuf,
|
pub exe_path: PathBuf,
|
||||||
|
@ -19,16 +24,15 @@ pub struct UpdateResult {
|
||||||
fn run_update(
|
fn run_update(
|
||||||
status: &JobContext,
|
status: &JobContext,
|
||||||
cancel: Receiver<()>,
|
cancel: Receiver<()>,
|
||||||
bin_name: String,
|
config: UpdateConfig,
|
||||||
) -> Result<Box<UpdateResult>> {
|
) -> Result<Box<UpdateResult>> {
|
||||||
update_status(status, "Fetching latest release".to_string(), 0, 3, &cancel)?;
|
update_status(status, "Fetching latest release".to_string(), 0, 3, &cancel)?;
|
||||||
let updater = build_updater().context("Failed to create release updater")?;
|
let updater = (config.build_updater)().context("Failed to create release updater")?;
|
||||||
let latest_release = updater.get_latest_release()?;
|
let latest_release = updater.get_latest_release()?;
|
||||||
let asset = latest_release
|
let asset =
|
||||||
.assets
|
latest_release.assets.iter().find(|a| a.name == config.bin_name).ok_or_else(|| {
|
||||||
.iter()
|
anyhow::Error::msg(format!("No release asset for {}", config.bin_name))
|
||||||
.find(|a| a.name == bin_name)
|
})?;
|
||||||
.ok_or_else(|| anyhow::Error::msg(format!("No release asset for {bin_name}")))?;
|
|
||||||
|
|
||||||
update_status(status, "Downloading release".to_string(), 1, 3, &cancel)?;
|
update_status(status, "Downloading release".to_string(), 1, 3, &cancel)?;
|
||||||
let tmp_dir = tempfile::Builder::new().prefix("update").tempdir_in(current_dir()?)?;
|
let tmp_dir = tempfile::Builder::new().prefix("update").tempdir_in(current_dir()?)?;
|
||||||
|
@ -47,9 +51,7 @@ fn run_update(
|
||||||
#[cfg(unix)]
|
#[cfg(unix)]
|
||||||
{
|
{
|
||||||
use std::{fs, os::unix::fs::PermissionsExt};
|
use std::{fs, os::unix::fs::PermissionsExt};
|
||||||
let mut perms = fs::metadata(&target_file)?.permissions();
|
fs::set_permissions(&target_file, fs::Permissions::from_mode(0o755))?;
|
||||||
perms.set_mode(0o755);
|
|
||||||
fs::set_permissions(&target_file, perms)?;
|
|
||||||
}
|
}
|
||||||
tmp_dir.close()?;
|
tmp_dir.close()?;
|
||||||
|
|
||||||
|
@ -57,8 +59,8 @@ fn run_update(
|
||||||
Ok(Box::from(UpdateResult { exe_path: target_file }))
|
Ok(Box::from(UpdateResult { exe_path: target_file }))
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn start_update(ctx: &egui::Context, bin_name: String) -> JobState {
|
pub fn start_update(waker: Waker, config: UpdateConfig) -> JobState {
|
||||||
start_job(ctx, "Update app", Job::Update, move |context, cancel| {
|
start_job(waker, "Update app", Job::Update, move |context, cancel| {
|
||||||
run_update(&context, cancel, bin_name).map(JobResult::Update)
|
run_update(&context, cancel, config).map(JobResult::Update)
|
||||||
})
|
})
|
||||||
}
|
}
|
|
@ -2,10 +2,14 @@
|
||||||
pub mod arch;
|
pub mod arch;
|
||||||
#[cfg(feature = "bindings")]
|
#[cfg(feature = "bindings")]
|
||||||
pub mod bindings;
|
pub mod bindings;
|
||||||
|
#[cfg(feature = "build")]
|
||||||
|
pub mod build;
|
||||||
#[cfg(feature = "config")]
|
#[cfg(feature = "config")]
|
||||||
pub mod config;
|
pub mod config;
|
||||||
#[cfg(feature = "any-arch")]
|
#[cfg(feature = "any-arch")]
|
||||||
pub mod diff;
|
pub mod diff;
|
||||||
|
#[cfg(feature = "build")]
|
||||||
|
pub mod jobs;
|
||||||
#[cfg(feature = "any-arch")]
|
#[cfg(feature = "any-arch")]
|
||||||
pub mod obj;
|
pub mod obj;
|
||||||
#[cfg(feature = "any-arch")]
|
#[cfg(feature = "any-arch")]
|
||||||
|
|
|
@ -38,7 +38,6 @@ float-ord = "0.3"
|
||||||
font-kit = "0.14"
|
font-kit = "0.14"
|
||||||
globset = { version = "0.4", features = ["serde1"] }
|
globset = { version = "0.4", features = ["serde1"] }
|
||||||
log = "0.4"
|
log = "0.4"
|
||||||
notify = { git = "https://github.com/notify-rs/notify", rev = "128bf6230c03d39dbb7f301ff7b20e594e34c3a2" }
|
|
||||||
objdiff-core = { path = "../objdiff-core", features = ["all"] }
|
objdiff-core = { path = "../objdiff-core", features = ["all"] }
|
||||||
open = "5.3"
|
open = "5.3"
|
||||||
png = "0.17"
|
png = "0.17"
|
||||||
|
@ -51,7 +50,6 @@ serde = { version = "1.0", features = ["derive"] }
|
||||||
serde_json = "1.0"
|
serde_json = "1.0"
|
||||||
shell-escape = "0.1"
|
shell-escape = "0.1"
|
||||||
strum = { version = "0.26", features = ["derive"] }
|
strum = { version = "0.26", features = ["derive"] }
|
||||||
tempfile = "3.14"
|
|
||||||
time = { version = "0.3", features = ["formatting", "local-offset"] }
|
time = { version = "0.3", features = ["formatting", "local-offset"] }
|
||||||
|
|
||||||
# Keep version in sync with egui
|
# Keep version in sync with egui
|
||||||
|
@ -76,18 +74,7 @@ features = [
|
||||||
optional = true
|
optional = true
|
||||||
default-features = false
|
default-features = false
|
||||||
|
|
||||||
# For Linux static binaries, use rustls
|
|
||||||
[target.'cfg(target_os = "linux")'.dependencies]
|
|
||||||
reqwest = { version = "0.12", default-features = false, features = ["blocking", "json", "multipart", "rustls-tls"] }
|
|
||||||
self_update = { version = "0.41", default-features = false, features = ["rustls"] }
|
|
||||||
|
|
||||||
# For all other platforms, use native TLS
|
|
||||||
[target.'cfg(not(target_os = "linux"))'.dependencies]
|
|
||||||
reqwest = { version = "0.12", default-features = false, features = ["blocking", "json", "multipart", "default-tls"] }
|
|
||||||
self_update = "0.41"
|
|
||||||
|
|
||||||
[target.'cfg(windows)'.dependencies]
|
[target.'cfg(windows)'.dependencies]
|
||||||
path-slash = "0.2"
|
|
||||||
winapi = "0.3"
|
winapi = "0.3"
|
||||||
|
|
||||||
[target.'cfg(unix)'.dependencies]
|
[target.'cfg(unix)'.dependencies]
|
||||||
|
|
|
@ -11,24 +11,22 @@ use std::{
|
||||||
};
|
};
|
||||||
|
|
||||||
use filetime::FileTime;
|
use filetime::FileTime;
|
||||||
use globset::{Glob, GlobSet};
|
use globset::Glob;
|
||||||
use notify::{RecursiveMode, Watcher};
|
|
||||||
use objdiff_core::{
|
use objdiff_core::{
|
||||||
|
build::watcher::{create_watcher, Watcher},
|
||||||
config::{
|
config::{
|
||||||
build_globset, save_project_config, ProjectConfig, ProjectConfigInfo, ProjectObject,
|
build_globset, default_watch_patterns, save_project_config, ProjectConfig,
|
||||||
ScratchConfig, SymbolMappings, DEFAULT_WATCH_PATTERNS,
|
ProjectConfigInfo, ProjectObject, ScratchConfig, SymbolMappings, DEFAULT_WATCH_PATTERNS,
|
||||||
},
|
},
|
||||||
diff::DiffObjConfig,
|
diff::DiffObjConfig,
|
||||||
|
jobs::{Job, JobQueue, JobResult},
|
||||||
};
|
};
|
||||||
use time::UtcOffset;
|
use time::UtcOffset;
|
||||||
|
|
||||||
use crate::{
|
use crate::{
|
||||||
app_config::{deserialize_config, AppConfigVersion},
|
app_config::{deserialize_config, AppConfigVersion},
|
||||||
config::{load_project_config, ProjectObjectNode},
|
config::{load_project_config, ProjectObjectNode},
|
||||||
jobs::{
|
jobs::{create_objdiff_config, egui_waker, start_build},
|
||||||
objdiff::{start_build, ObjDiffConfig},
|
|
||||||
Job, JobQueue, JobResult, JobStatus,
|
|
||||||
},
|
|
||||||
views::{
|
views::{
|
||||||
appearance::{appearance_window, Appearance},
|
appearance::{appearance_window, Appearance},
|
||||||
config::{
|
config::{
|
||||||
|
@ -121,11 +119,6 @@ impl From<&ProjectObject> for ObjectConfig {
|
||||||
#[inline]
|
#[inline]
|
||||||
fn bool_true() -> bool { true }
|
fn bool_true() -> bool { true }
|
||||||
|
|
||||||
#[inline]
|
|
||||||
fn default_watch_patterns() -> Vec<Glob> {
|
|
||||||
DEFAULT_WATCH_PATTERNS.iter().map(|s| Glob::new(s).unwrap()).collect()
|
|
||||||
}
|
|
||||||
|
|
||||||
pub struct AppState {
|
pub struct AppState {
|
||||||
pub config: AppConfig,
|
pub config: AppConfig,
|
||||||
pub objects: Vec<ProjectObject>,
|
pub objects: Vec<ProjectObject>,
|
||||||
|
@ -399,7 +392,7 @@ pub struct App {
|
||||||
view_state: ViewState,
|
view_state: ViewState,
|
||||||
state: AppStateRef,
|
state: AppStateRef,
|
||||||
modified: Arc<AtomicBool>,
|
modified: Arc<AtomicBool>,
|
||||||
watcher: Option<notify::RecommendedWatcher>,
|
watcher: Option<Watcher>,
|
||||||
app_path: Option<PathBuf>,
|
app_path: Option<PathBuf>,
|
||||||
relaunch_path: Rc<Mutex<Option<PathBuf>>>,
|
relaunch_path: Rc<Mutex<Option<PathBuf>>>,
|
||||||
should_relaunch: bool,
|
should_relaunch: bool,
|
||||||
|
@ -474,53 +467,17 @@ impl App {
|
||||||
|
|
||||||
let ViewState { jobs, diff_state, config_state, .. } = &mut self.view_state;
|
let ViewState { jobs, diff_state, config_state, .. } = &mut self.view_state;
|
||||||
|
|
||||||
let mut results = vec![];
|
jobs.collect_results();
|
||||||
for (job, result) in jobs.iter_finished() {
|
jobs.results.retain(|result| match result {
|
||||||
match result {
|
|
||||||
Ok(result) => {
|
|
||||||
log::info!("Job {} finished", job.id);
|
|
||||||
match result {
|
|
||||||
JobResult::None => {
|
|
||||||
if let Some(err) = &job.context.status.read().unwrap().error {
|
|
||||||
log::error!("{:?}", err);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
JobResult::Update(state) => {
|
JobResult::Update(state) => {
|
||||||
if let Ok(mut guard) = self.relaunch_path.lock() {
|
if let Ok(mut guard) = self.relaunch_path.lock() {
|
||||||
*guard = Some(state.exe_path);
|
*guard = Some(state.exe_path.clone());
|
||||||
self.should_relaunch = true;
|
self.should_relaunch = true;
|
||||||
}
|
}
|
||||||
|
false
|
||||||
}
|
}
|
||||||
_ => results.push(result),
|
_ => true,
|
||||||
}
|
});
|
||||||
}
|
|
||||||
Err(err) => {
|
|
||||||
let err = if let Some(msg) = err.downcast_ref::<&'static str>() {
|
|
||||||
anyhow::Error::msg(*msg)
|
|
||||||
} else if let Some(msg) = err.downcast_ref::<String>() {
|
|
||||||
anyhow::Error::msg(msg.clone())
|
|
||||||
} else {
|
|
||||||
anyhow::Error::msg("Thread panicked")
|
|
||||||
};
|
|
||||||
let result = job.context.status.write();
|
|
||||||
if let Ok(mut guard) = result {
|
|
||||||
guard.error = Some(err);
|
|
||||||
} else {
|
|
||||||
drop(result);
|
|
||||||
job.context.status = Arc::new(RwLock::new(JobStatus {
|
|
||||||
title: "Error".to_string(),
|
|
||||||
progress_percent: 0.0,
|
|
||||||
progress_items: None,
|
|
||||||
status: String::new(),
|
|
||||||
error: Some(err),
|
|
||||||
}));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
jobs.results.append(&mut results);
|
|
||||||
jobs.clear_finished();
|
|
||||||
|
|
||||||
diff_state.pre_update(jobs, &self.state);
|
diff_state.pre_update(jobs, &self.state);
|
||||||
config_state.pre_update(jobs, &self.state);
|
config_state.pre_update(jobs, &self.state);
|
||||||
debug_assert!(jobs.results.is_empty());
|
debug_assert!(jobs.results.is_empty());
|
||||||
|
@ -572,7 +529,7 @@ impl App {
|
||||||
match build_globset(&state.config.watch_patterns)
|
match build_globset(&state.config.watch_patterns)
|
||||||
.map_err(anyhow::Error::new)
|
.map_err(anyhow::Error::new)
|
||||||
.and_then(|globset| {
|
.and_then(|globset| {
|
||||||
create_watcher(ctx.clone(), self.modified.clone(), project_dir, globset)
|
create_watcher(self.modified.clone(), project_dir, globset, egui_waker(ctx))
|
||||||
.map_err(anyhow::Error::new)
|
.map_err(anyhow::Error::new)
|
||||||
}) {
|
}) {
|
||||||
Ok(watcher) => self.watcher = Some(watcher),
|
Ok(watcher) => self.watcher = Some(watcher),
|
||||||
|
@ -619,15 +576,15 @@ impl App {
|
||||||
&& state.config.selected_obj.is_some()
|
&& state.config.selected_obj.is_some()
|
||||||
&& !jobs.is_running(Job::ObjDiff)
|
&& !jobs.is_running(Job::ObjDiff)
|
||||||
{
|
{
|
||||||
jobs.push(start_build(ctx, ObjDiffConfig::from_state(state)));
|
start_build(ctx, jobs, create_objdiff_config(state));
|
||||||
state.queue_build = false;
|
state.queue_build = false;
|
||||||
state.queue_reload = false;
|
state.queue_reload = false;
|
||||||
} else if state.queue_reload && !jobs.is_running(Job::ObjDiff) {
|
} else if state.queue_reload && !jobs.is_running(Job::ObjDiff) {
|
||||||
let mut diff_config = ObjDiffConfig::from_state(state);
|
let mut diff_config = create_objdiff_config(state);
|
||||||
// Don't build, just reload the current files
|
// Don't build, just reload the current files
|
||||||
diff_config.build_base = false;
|
diff_config.build_base = false;
|
||||||
diff_config.build_target = false;
|
diff_config.build_target = false;
|
||||||
jobs.push(start_build(ctx, diff_config));
|
start_build(ctx, jobs, diff_config);
|
||||||
state.queue_reload = false;
|
state.queue_reload = false;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -854,40 +811,6 @@ impl eframe::App for App {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
fn create_watcher(
|
|
||||||
ctx: egui::Context,
|
|
||||||
modified: Arc<AtomicBool>,
|
|
||||||
project_dir: &Path,
|
|
||||||
patterns: GlobSet,
|
|
||||||
) -> notify::Result<notify::RecommendedWatcher> {
|
|
||||||
let base_dir = project_dir.to_owned();
|
|
||||||
let mut watcher =
|
|
||||||
notify::recommended_watcher(move |res: notify::Result<notify::Event>| match res {
|
|
||||||
Ok(event) => {
|
|
||||||
if matches!(
|
|
||||||
event.kind,
|
|
||||||
notify::EventKind::Modify(..)
|
|
||||||
| notify::EventKind::Create(..)
|
|
||||||
| notify::EventKind::Remove(..)
|
|
||||||
) {
|
|
||||||
for path in &event.paths {
|
|
||||||
let Ok(path) = path.strip_prefix(&base_dir) else {
|
|
||||||
continue;
|
|
||||||
};
|
|
||||||
if patterns.is_match(path) {
|
|
||||||
log::info!("File modified: {}", path.display());
|
|
||||||
modified.store(true, Ordering::Relaxed);
|
|
||||||
ctx.request_repaint();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
Err(e) => log::error!("watch error: {e:?}"),
|
|
||||||
})?;
|
|
||||||
watcher.watch(project_dir, RecursiveMode::Recursive)?;
|
|
||||||
Ok(watcher)
|
|
||||||
}
|
|
||||||
|
|
||||||
#[inline]
|
#[inline]
|
||||||
fn file_modified(path: &Path, last_ts: FileTime) -> bool {
|
fn file_modified(path: &Path, last_ts: FileTime) -> bool {
|
||||||
if let Ok(metadata) = fs::metadata(path) {
|
if let Ok(metadata) = fs::metadata(path) {
|
||||||
|
|
|
@ -0,0 +1,138 @@
|
||||||
|
use std::{
|
||||||
|
sync::Arc,
|
||||||
|
task::{Wake, Waker},
|
||||||
|
};
|
||||||
|
|
||||||
|
use anyhow::{bail, Result};
|
||||||
|
use jobs::create_scratch;
|
||||||
|
use objdiff_core::{
|
||||||
|
build::BuildConfig,
|
||||||
|
jobs,
|
||||||
|
jobs::{check_update::CheckUpdateConfig, objdiff, update::UpdateConfig, Job, JobQueue},
|
||||||
|
};
|
||||||
|
|
||||||
|
use crate::{
|
||||||
|
app::{AppConfig, AppState},
|
||||||
|
update::{build_updater, BIN_NAME_NEW, BIN_NAME_OLD},
|
||||||
|
};
|
||||||
|
|
||||||
|
struct EguiWaker(egui::Context);
|
||||||
|
|
||||||
|
impl Wake for EguiWaker {
|
||||||
|
fn wake(self: Arc<Self>) { self.0.request_repaint(); }
|
||||||
|
|
||||||
|
fn wake_by_ref(self: &Arc<Self>) { self.0.request_repaint(); }
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn egui_waker(ctx: &egui::Context) -> Waker { Waker::from(Arc::new(EguiWaker(ctx.clone()))) }
|
||||||
|
|
||||||
|
pub fn is_create_scratch_available(config: &AppConfig) -> bool {
|
||||||
|
let Some(selected_obj) = &config.selected_obj else {
|
||||||
|
return false;
|
||||||
|
};
|
||||||
|
selected_obj.target_path.is_some() && selected_obj.scratch.is_some()
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn start_create_scratch(
|
||||||
|
ctx: &egui::Context,
|
||||||
|
jobs: &mut JobQueue,
|
||||||
|
state: &AppState,
|
||||||
|
function_name: String,
|
||||||
|
) {
|
||||||
|
match create_scratch_config(state, function_name) {
|
||||||
|
Ok(config) => {
|
||||||
|
jobs.push_once(Job::CreateScratch, || {
|
||||||
|
create_scratch::start_create_scratch(egui_waker(ctx), config)
|
||||||
|
});
|
||||||
|
}
|
||||||
|
Err(err) => {
|
||||||
|
log::error!("Failed to create scratch config: {err}");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn create_scratch_config(
|
||||||
|
state: &AppState,
|
||||||
|
function_name: String,
|
||||||
|
) -> Result<create_scratch::CreateScratchConfig> {
|
||||||
|
let Some(selected_obj) = &state.config.selected_obj else {
|
||||||
|
bail!("No object selected");
|
||||||
|
};
|
||||||
|
let Some(target_path) = &selected_obj.target_path else {
|
||||||
|
bail!("No target path for {}", selected_obj.name);
|
||||||
|
};
|
||||||
|
let Some(scratch_config) = &selected_obj.scratch else {
|
||||||
|
bail!("No scratch configuration for {}", selected_obj.name);
|
||||||
|
};
|
||||||
|
Ok(create_scratch::CreateScratchConfig {
|
||||||
|
build_config: BuildConfig::from(&state.config),
|
||||||
|
context_path: scratch_config.ctx_path.clone(),
|
||||||
|
build_context: scratch_config.build_ctx.unwrap_or(false),
|
||||||
|
compiler: scratch_config.compiler.clone().unwrap_or_default(),
|
||||||
|
platform: scratch_config.platform.clone().unwrap_or_default(),
|
||||||
|
compiler_flags: scratch_config.c_flags.clone().unwrap_or_default(),
|
||||||
|
function_name,
|
||||||
|
target_obj: target_path.to_path_buf(),
|
||||||
|
preset_id: scratch_config.preset_id,
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
impl From<&AppConfig> for BuildConfig {
|
||||||
|
fn from(config: &AppConfig) -> Self {
|
||||||
|
Self {
|
||||||
|
project_dir: config.project_dir.clone(),
|
||||||
|
custom_make: config.custom_make.clone(),
|
||||||
|
custom_args: config.custom_args.clone(),
|
||||||
|
selected_wsl_distro: config.selected_wsl_distro.clone(),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn create_objdiff_config(state: &AppState) -> objdiff::ObjDiffConfig {
|
||||||
|
objdiff::ObjDiffConfig {
|
||||||
|
build_config: BuildConfig::from(&state.config),
|
||||||
|
build_base: state.config.build_base,
|
||||||
|
build_target: state.config.build_target,
|
||||||
|
target_path: state
|
||||||
|
.config
|
||||||
|
.selected_obj
|
||||||
|
.as_ref()
|
||||||
|
.and_then(|obj| obj.target_path.as_ref())
|
||||||
|
.cloned(),
|
||||||
|
base_path: state
|
||||||
|
.config
|
||||||
|
.selected_obj
|
||||||
|
.as_ref()
|
||||||
|
.and_then(|obj| obj.base_path.as_ref())
|
||||||
|
.cloned(),
|
||||||
|
diff_obj_config: state.config.diff_obj_config.clone(),
|
||||||
|
symbol_mappings: state
|
||||||
|
.config
|
||||||
|
.selected_obj
|
||||||
|
.as_ref()
|
||||||
|
.map(|obj| &obj.symbol_mappings)
|
||||||
|
.cloned()
|
||||||
|
.unwrap_or_default(),
|
||||||
|
selecting_left: state.selecting_left.clone(),
|
||||||
|
selecting_right: state.selecting_right.clone(),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn start_build(ctx: &egui::Context, jobs: &mut JobQueue, config: objdiff::ObjDiffConfig) {
|
||||||
|
jobs.push_once(Job::ObjDiff, || objdiff::start_build(egui_waker(ctx), config));
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn start_check_update(ctx: &egui::Context, jobs: &mut JobQueue) {
|
||||||
|
jobs.push_once(Job::Update, || {
|
||||||
|
jobs::check_update::start_check_update(egui_waker(ctx), CheckUpdateConfig {
|
||||||
|
build_updater,
|
||||||
|
bin_names: vec![BIN_NAME_NEW.to_string(), BIN_NAME_OLD.to_string()],
|
||||||
|
})
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn start_update(ctx: &egui::Context, jobs: &mut JobQueue, bin_name: String) {
|
||||||
|
jobs.push_once(Job::Update, || {
|
||||||
|
jobs::update::start_update(egui_waker(ctx), UpdateConfig { build_updater, bin_name })
|
||||||
|
});
|
||||||
|
}
|
|
@ -1,39 +0,0 @@
|
||||||
use std::sync::mpsc::Receiver;
|
|
||||||
|
|
||||||
use anyhow::{Context, Result};
|
|
||||||
use self_update::{cargo_crate_version, update::Release};
|
|
||||||
|
|
||||||
use crate::{
|
|
||||||
jobs::{start_job, update_status, Job, JobContext, JobResult, JobState},
|
|
||||||
update::{build_updater, BIN_NAME_NEW, BIN_NAME_OLD},
|
|
||||||
};
|
|
||||||
|
|
||||||
pub struct CheckUpdateResult {
|
|
||||||
pub update_available: bool,
|
|
||||||
pub latest_release: Release,
|
|
||||||
pub found_binary: Option<String>,
|
|
||||||
}
|
|
||||||
|
|
||||||
fn run_check_update(context: &JobContext, cancel: Receiver<()>) -> Result<Box<CheckUpdateResult>> {
|
|
||||||
update_status(context, "Fetching latest release".to_string(), 0, 1, &cancel)?;
|
|
||||||
let updater = build_updater().context("Failed to create release updater")?;
|
|
||||||
let latest_release = updater.get_latest_release()?;
|
|
||||||
let update_available =
|
|
||||||
self_update::version::bump_is_greater(cargo_crate_version!(), &latest_release.version)?;
|
|
||||||
// Find the binary name in the release assets
|
|
||||||
let found_binary = latest_release
|
|
||||||
.assets
|
|
||||||
.iter()
|
|
||||||
.find(|a| a.name == BIN_NAME_NEW)
|
|
||||||
.or_else(|| latest_release.assets.iter().find(|a| a.name == BIN_NAME_OLD))
|
|
||||||
.map(|a| a.name.clone());
|
|
||||||
|
|
||||||
update_status(context, "Complete".to_string(), 1, 1, &cancel)?;
|
|
||||||
Ok(Box::new(CheckUpdateResult { update_available, latest_release, found_binary }))
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn start_check_update(ctx: &egui::Context) -> JobState {
|
|
||||||
start_job(ctx, "Check for updates", Job::CheckUpdate, move |context, cancel| {
|
|
||||||
run_check_update(&context, cancel).map(|result| JobResult::CheckUpdate(Some(result)))
|
|
||||||
})
|
|
||||||
}
|
|
|
@ -1,328 +0,0 @@
|
||||||
use std::{
|
|
||||||
path::{Path, PathBuf},
|
|
||||||
process::Command,
|
|
||||||
sync::mpsc::Receiver,
|
|
||||||
};
|
|
||||||
|
|
||||||
use anyhow::{anyhow, Error, Result};
|
|
||||||
use objdiff_core::{
|
|
||||||
diff::{diff_objs, DiffObjConfig, MappingConfig, ObjDiff},
|
|
||||||
obj::{read, ObjInfo},
|
|
||||||
};
|
|
||||||
use time::OffsetDateTime;
|
|
||||||
|
|
||||||
use crate::{
|
|
||||||
app::{AppConfig, AppState, ObjectConfig},
|
|
||||||
jobs::{start_job, update_status, Job, JobContext, JobResult, JobState},
|
|
||||||
};
|
|
||||||
|
|
||||||
pub struct BuildStatus {
|
|
||||||
pub success: bool,
|
|
||||||
pub cmdline: String,
|
|
||||||
pub stdout: String,
|
|
||||||
pub stderr: String,
|
|
||||||
}
|
|
||||||
|
|
||||||
impl Default for BuildStatus {
|
|
||||||
fn default() -> Self {
|
|
||||||
BuildStatus {
|
|
||||||
success: true,
|
|
||||||
cmdline: String::new(),
|
|
||||||
stdout: String::new(),
|
|
||||||
stderr: String::new(),
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
#[derive(Debug, Clone)]
|
|
||||||
pub struct BuildConfig {
|
|
||||||
pub project_dir: Option<PathBuf>,
|
|
||||||
pub custom_make: Option<String>,
|
|
||||||
pub custom_args: Option<Vec<String>>,
|
|
||||||
#[allow(unused)]
|
|
||||||
pub selected_wsl_distro: Option<String>,
|
|
||||||
}
|
|
||||||
|
|
||||||
impl BuildConfig {
|
|
||||||
pub(crate) fn from_config(config: &AppConfig) -> Self {
|
|
||||||
Self {
|
|
||||||
project_dir: config.project_dir.clone(),
|
|
||||||
custom_make: config.custom_make.clone(),
|
|
||||||
custom_args: config.custom_args.clone(),
|
|
||||||
selected_wsl_distro: config.selected_wsl_distro.clone(),
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
pub struct ObjDiffConfig {
|
|
||||||
pub build_config: BuildConfig,
|
|
||||||
pub build_base: bool,
|
|
||||||
pub build_target: bool,
|
|
||||||
pub selected_obj: Option<ObjectConfig>,
|
|
||||||
pub diff_obj_config: DiffObjConfig,
|
|
||||||
pub selecting_left: Option<String>,
|
|
||||||
pub selecting_right: Option<String>,
|
|
||||||
}
|
|
||||||
|
|
||||||
impl ObjDiffConfig {
|
|
||||||
pub(crate) fn from_state(state: &AppState) -> Self {
|
|
||||||
Self {
|
|
||||||
build_config: BuildConfig::from_config(&state.config),
|
|
||||||
build_base: state.config.build_base,
|
|
||||||
build_target: state.config.build_target,
|
|
||||||
selected_obj: state.config.selected_obj.clone(),
|
|
||||||
diff_obj_config: state.config.diff_obj_config.clone(),
|
|
||||||
selecting_left: state.selecting_left.clone(),
|
|
||||||
selecting_right: state.selecting_right.clone(),
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
pub struct ObjDiffResult {
|
|
||||||
pub first_status: BuildStatus,
|
|
||||||
pub second_status: BuildStatus,
|
|
||||||
pub first_obj: Option<(ObjInfo, ObjDiff)>,
|
|
||||||
pub second_obj: Option<(ObjInfo, ObjDiff)>,
|
|
||||||
pub time: OffsetDateTime,
|
|
||||||
}
|
|
||||||
|
|
||||||
pub(crate) fn run_make(config: &BuildConfig, arg: &Path) -> BuildStatus {
|
|
||||||
let Some(cwd) = &config.project_dir else {
|
|
||||||
return BuildStatus {
|
|
||||||
success: false,
|
|
||||||
stderr: "Missing project dir".to_string(),
|
|
||||||
..Default::default()
|
|
||||||
};
|
|
||||||
};
|
|
||||||
let make = config.custom_make.as_deref().unwrap_or("make");
|
|
||||||
let make_args = config.custom_args.as_deref().unwrap_or(&[]);
|
|
||||||
#[cfg(not(windows))]
|
|
||||||
let mut command = {
|
|
||||||
let mut command = Command::new(make);
|
|
||||||
command.current_dir(cwd).args(make_args).arg(arg);
|
|
||||||
command
|
|
||||||
};
|
|
||||||
#[cfg(windows)]
|
|
||||||
let mut command = {
|
|
||||||
use std::os::windows::process::CommandExt;
|
|
||||||
|
|
||||||
use path_slash::PathExt;
|
|
||||||
let mut command = if config.selected_wsl_distro.is_some() {
|
|
||||||
Command::new("wsl")
|
|
||||||
} else {
|
|
||||||
Command::new(make)
|
|
||||||
};
|
|
||||||
if let Some(distro) = &config.selected_wsl_distro {
|
|
||||||
// Strip distro root prefix \\wsl.localhost\{distro}
|
|
||||||
let wsl_path_prefix = format!("\\\\wsl.localhost\\{}", distro);
|
|
||||||
let cwd = match cwd.strip_prefix(wsl_path_prefix) {
|
|
||||||
Ok(new_cwd) => format!("/{}", new_cwd.to_slash_lossy().as_ref()),
|
|
||||||
Err(_) => cwd.to_string_lossy().to_string(),
|
|
||||||
};
|
|
||||||
|
|
||||||
command
|
|
||||||
.arg("--cd")
|
|
||||||
.arg(cwd)
|
|
||||||
.arg("-d")
|
|
||||||
.arg(distro)
|
|
||||||
.arg("--")
|
|
||||||
.arg(make)
|
|
||||||
.args(make_args)
|
|
||||||
.arg(arg.to_slash_lossy().as_ref());
|
|
||||||
} else {
|
|
||||||
command.current_dir(cwd).args(make_args).arg(arg.to_slash_lossy().as_ref());
|
|
||||||
}
|
|
||||||
command.creation_flags(winapi::um::winbase::CREATE_NO_WINDOW);
|
|
||||||
command
|
|
||||||
};
|
|
||||||
let mut cmdline = shell_escape::escape(command.get_program().to_string_lossy()).into_owned();
|
|
||||||
for arg in command.get_args() {
|
|
||||||
cmdline.push(' ');
|
|
||||||
cmdline.push_str(shell_escape::escape(arg.to_string_lossy()).as_ref());
|
|
||||||
}
|
|
||||||
let output = match command.output() {
|
|
||||||
Ok(output) => output,
|
|
||||||
Err(e) => {
|
|
||||||
return BuildStatus {
|
|
||||||
success: false,
|
|
||||||
cmdline,
|
|
||||||
stdout: Default::default(),
|
|
||||||
stderr: e.to_string(),
|
|
||||||
};
|
|
||||||
}
|
|
||||||
};
|
|
||||||
// Try from_utf8 first to avoid copying the buffer if it's valid, then fall back to from_utf8_lossy
|
|
||||||
let stdout = String::from_utf8(output.stdout)
|
|
||||||
.unwrap_or_else(|e| String::from_utf8_lossy(e.as_bytes()).into_owned());
|
|
||||||
let stderr = String::from_utf8(output.stderr)
|
|
||||||
.unwrap_or_else(|e| String::from_utf8_lossy(e.as_bytes()).into_owned());
|
|
||||||
BuildStatus { success: output.status.success(), cmdline, stdout, stderr }
|
|
||||||
}
|
|
||||||
|
|
||||||
fn run_build(
|
|
||||||
context: &JobContext,
|
|
||||||
cancel: Receiver<()>,
|
|
||||||
mut config: ObjDiffConfig,
|
|
||||||
) -> Result<Box<ObjDiffResult>> {
|
|
||||||
let obj_config = config.selected_obj.ok_or_else(|| Error::msg("Missing obj path"))?;
|
|
||||||
// Use the per-object symbol mappings, we don't set mappings globally
|
|
||||||
config.diff_obj_config.symbol_mappings = MappingConfig {
|
|
||||||
mappings: obj_config.symbol_mappings,
|
|
||||||
selecting_left: config.selecting_left,
|
|
||||||
selecting_right: config.selecting_right,
|
|
||||||
};
|
|
||||||
|
|
||||||
let project_dir = config
|
|
||||||
.build_config
|
|
||||||
.project_dir
|
|
||||||
.as_ref()
|
|
||||||
.ok_or_else(|| Error::msg("Missing project dir"))?;
|
|
||||||
let target_path_rel = if let Some(target_path) = &obj_config.target_path {
|
|
||||||
Some(target_path.strip_prefix(project_dir).map_err(|_| {
|
|
||||||
anyhow!(
|
|
||||||
"Target path '{}' doesn't begin with '{}'",
|
|
||||||
target_path.display(),
|
|
||||||
project_dir.display()
|
|
||||||
)
|
|
||||||
})?)
|
|
||||||
} else {
|
|
||||||
None
|
|
||||||
};
|
|
||||||
let base_path_rel = if let Some(base_path) = &obj_config.base_path {
|
|
||||||
Some(base_path.strip_prefix(project_dir).map_err(|_| {
|
|
||||||
anyhow!(
|
|
||||||
"Base path '{}' doesn't begin with '{}'",
|
|
||||||
base_path.display(),
|
|
||||||
project_dir.display()
|
|
||||||
)
|
|
||||||
})?)
|
|
||||||
} else {
|
|
||||||
None
|
|
||||||
};
|
|
||||||
|
|
||||||
let mut total = 1;
|
|
||||||
if config.build_target && target_path_rel.is_some() {
|
|
||||||
total += 1;
|
|
||||||
}
|
|
||||||
if config.build_base && base_path_rel.is_some() {
|
|
||||||
total += 1;
|
|
||||||
}
|
|
||||||
if target_path_rel.is_some() {
|
|
||||||
total += 1;
|
|
||||||
}
|
|
||||||
if base_path_rel.is_some() {
|
|
||||||
total += 1;
|
|
||||||
}
|
|
||||||
|
|
||||||
let mut step_idx = 0;
|
|
||||||
let mut first_status = match target_path_rel {
|
|
||||||
Some(target_path_rel) if config.build_target => {
|
|
||||||
update_status(
|
|
||||||
context,
|
|
||||||
format!("Building target {}", target_path_rel.display()),
|
|
||||||
step_idx,
|
|
||||||
total,
|
|
||||||
&cancel,
|
|
||||||
)?;
|
|
||||||
step_idx += 1;
|
|
||||||
run_make(&config.build_config, target_path_rel)
|
|
||||||
}
|
|
||||||
_ => BuildStatus::default(),
|
|
||||||
};
|
|
||||||
|
|
||||||
let mut second_status = match base_path_rel {
|
|
||||||
Some(base_path_rel) if config.build_base => {
|
|
||||||
update_status(
|
|
||||||
context,
|
|
||||||
format!("Building base {}", base_path_rel.display()),
|
|
||||||
step_idx,
|
|
||||||
total,
|
|
||||||
&cancel,
|
|
||||||
)?;
|
|
||||||
step_idx += 1;
|
|
||||||
run_make(&config.build_config, base_path_rel)
|
|
||||||
}
|
|
||||||
_ => BuildStatus::default(),
|
|
||||||
};
|
|
||||||
|
|
||||||
let time = OffsetDateTime::now_utc();
|
|
||||||
|
|
||||||
let first_obj = match &obj_config.target_path {
|
|
||||||
Some(target_path) if first_status.success => {
|
|
||||||
update_status(
|
|
||||||
context,
|
|
||||||
format!("Loading target {}", target_path_rel.unwrap().display()),
|
|
||||||
step_idx,
|
|
||||||
total,
|
|
||||||
&cancel,
|
|
||||||
)?;
|
|
||||||
step_idx += 1;
|
|
||||||
match read::read(target_path, &config.diff_obj_config) {
|
|
||||||
Ok(obj) => Some(obj),
|
|
||||||
Err(e) => {
|
|
||||||
first_status = BuildStatus {
|
|
||||||
success: false,
|
|
||||||
stdout: format!("Loading object '{}'", target_path.display()),
|
|
||||||
stderr: format!("{:#}", e),
|
|
||||||
..Default::default()
|
|
||||||
};
|
|
||||||
None
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
Some(_) => {
|
|
||||||
step_idx += 1;
|
|
||||||
None
|
|
||||||
}
|
|
||||||
_ => None,
|
|
||||||
};
|
|
||||||
|
|
||||||
let second_obj = match &obj_config.base_path {
|
|
||||||
Some(base_path) if second_status.success => {
|
|
||||||
update_status(
|
|
||||||
context,
|
|
||||||
format!("Loading base {}", base_path_rel.unwrap().display()),
|
|
||||||
step_idx,
|
|
||||||
total,
|
|
||||||
&cancel,
|
|
||||||
)?;
|
|
||||||
step_idx += 1;
|
|
||||||
match read::read(base_path, &config.diff_obj_config) {
|
|
||||||
Ok(obj) => Some(obj),
|
|
||||||
Err(e) => {
|
|
||||||
second_status = BuildStatus {
|
|
||||||
success: false,
|
|
||||||
stdout: format!("Loading object '{}'", base_path.display()),
|
|
||||||
stderr: format!("{:#}", e),
|
|
||||||
..Default::default()
|
|
||||||
};
|
|
||||||
None
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
Some(_) => {
|
|
||||||
step_idx += 1;
|
|
||||||
None
|
|
||||||
}
|
|
||||||
_ => None,
|
|
||||||
};
|
|
||||||
|
|
||||||
update_status(context, "Performing diff".to_string(), step_idx, total, &cancel)?;
|
|
||||||
step_idx += 1;
|
|
||||||
let result = diff_objs(&config.diff_obj_config, first_obj.as_ref(), second_obj.as_ref(), None)?;
|
|
||||||
|
|
||||||
update_status(context, "Complete".to_string(), step_idx, total, &cancel)?;
|
|
||||||
Ok(Box::new(ObjDiffResult {
|
|
||||||
first_status,
|
|
||||||
second_status,
|
|
||||||
first_obj: first_obj.and_then(|o| result.left.map(|d| (o, d))),
|
|
||||||
second_obj: second_obj.and_then(|o| result.right.map(|d| (o, d))),
|
|
||||||
time,
|
|
||||||
}))
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn start_build(ctx: &egui::Context, config: ObjDiffConfig) -> JobState {
|
|
||||||
start_job(ctx, "Build", Job::ObjDiff, move |context, cancel| {
|
|
||||||
run_build(&context, cancel, config).map(|result| JobResult::ObjDiff(Some(result)))
|
|
||||||
})
|
|
||||||
}
|
|
|
@ -1,5 +1,7 @@
|
||||||
|
use anyhow::Result;
|
||||||
use cfg_if::cfg_if;
|
use cfg_if::cfg_if;
|
||||||
use const_format::formatcp;
|
use const_format::formatcp;
|
||||||
|
use objdiff_core::jobs::update::self_update;
|
||||||
use self_update::{cargo_crate_version, update::ReleaseUpdate};
|
use self_update::{cargo_crate_version, update::ReleaseUpdate};
|
||||||
|
|
||||||
pub const OS: &str = std::env::consts::OS;
|
pub const OS: &str = std::env::consts::OS;
|
||||||
|
@ -26,8 +28,8 @@ pub const BIN_NAME_OLD: &str = formatcp!("objdiff-{}-{}{}", OS, ARCH, std::env::
|
||||||
pub const RELEASE_URL: &str =
|
pub const RELEASE_URL: &str =
|
||||||
formatcp!("https://github.com/{}/{}/releases/latest", GITHUB_USER, GITHUB_REPO);
|
formatcp!("https://github.com/{}/{}/releases/latest", GITHUB_USER, GITHUB_REPO);
|
||||||
|
|
||||||
pub fn build_updater() -> self_update::errors::Result<Box<dyn ReleaseUpdate>> {
|
pub fn build_updater() -> Result<Box<dyn ReleaseUpdate>> {
|
||||||
self_update::backends::github::Update::configure()
|
Ok(self_update::backends::github::Update::configure()
|
||||||
.repo_owner(GITHUB_USER)
|
.repo_owner(GITHUB_USER)
|
||||||
.repo_name(GITHUB_REPO)
|
.repo_name(GITHUB_REPO)
|
||||||
// bin_name is required, but unused?
|
// bin_name is required, but unused?
|
||||||
|
@ -35,5 +37,5 @@ pub fn build_updater() -> self_update::errors::Result<Box<dyn ReleaseUpdate>> {
|
||||||
.no_confirm(true)
|
.no_confirm(true)
|
||||||
.show_output(false)
|
.show_output(false)
|
||||||
.current_version(cargo_crate_version!())
|
.current_version(cargo_crate_version!())
|
||||||
.build()
|
.build()?)
|
||||||
}
|
}
|
||||||
|
|
|
@ -15,6 +15,7 @@ use globset::Glob;
|
||||||
use objdiff_core::{
|
use objdiff_core::{
|
||||||
config::{ProjectObject, DEFAULT_WATCH_PATTERNS},
|
config::{ProjectObject, DEFAULT_WATCH_PATTERNS},
|
||||||
diff::{ArmArchVersion, ArmR9Usage, MipsAbi, MipsInstrCategory, X86Formatter},
|
diff::{ArmArchVersion, ArmR9Usage, MipsAbi, MipsInstrCategory, X86Formatter},
|
||||||
|
jobs::{check_update::CheckUpdateResult, Job, JobQueue, JobResult},
|
||||||
};
|
};
|
||||||
use strum::{EnumMessage, VariantArray};
|
use strum::{EnumMessage, VariantArray};
|
||||||
|
|
||||||
|
@ -22,11 +23,7 @@ use crate::{
|
||||||
app::{AppConfig, AppState, AppStateRef, ObjectConfig},
|
app::{AppConfig, AppState, AppStateRef, ObjectConfig},
|
||||||
config::ProjectObjectNode,
|
config::ProjectObjectNode,
|
||||||
hotkeys,
|
hotkeys,
|
||||||
jobs::{
|
jobs::{start_check_update, start_update},
|
||||||
check_update::{start_check_update, CheckUpdateResult},
|
|
||||||
update::start_update,
|
|
||||||
Job, JobQueue, JobResult,
|
|
||||||
},
|
|
||||||
update::RELEASE_URL,
|
update::RELEASE_URL,
|
||||||
views::{
|
views::{
|
||||||
appearance::Appearance,
|
appearance::Appearance,
|
||||||
|
@ -119,11 +116,11 @@ impl ConfigViewState {
|
||||||
|
|
||||||
if self.queue_check_update {
|
if self.queue_check_update {
|
||||||
self.queue_check_update = false;
|
self.queue_check_update = false;
|
||||||
jobs.push_once(Job::CheckUpdate, || start_check_update(ctx));
|
start_check_update(ctx, jobs);
|
||||||
}
|
}
|
||||||
|
|
||||||
if let Some(bin_name) = self.queue_update.take() {
|
if let Some(bin_name) = self.queue_update.take() {
|
||||||
jobs.push_once(Job::Update, || start_update(ctx, bin_name));
|
start_update(ctx, jobs, bin_name);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,11 +1,9 @@
|
||||||
use std::cmp::Ordering;
|
use std::cmp::Ordering;
|
||||||
|
|
||||||
use egui::{ProgressBar, RichText, Widget};
|
use egui::{ProgressBar, RichText, Widget};
|
||||||
|
use objdiff_core::jobs::{JobQueue, JobStatus};
|
||||||
|
|
||||||
use crate::{
|
use crate::views::appearance::Appearance;
|
||||||
jobs::{JobQueue, JobStatus},
|
|
||||||
views::appearance::Appearance,
|
|
||||||
};
|
|
||||||
|
|
||||||
pub fn jobs_ui(ui: &mut egui::Ui, jobs: &mut JobQueue, appearance: &Appearance) {
|
pub fn jobs_ui(ui: &mut egui::Ui, jobs: &mut JobQueue, appearance: &Appearance) {
|
||||||
if ui.button("Clear").clicked() {
|
if ui.button("Clear").clicked() {
|
||||||
|
|
|
@ -6,7 +6,9 @@ use egui::{
|
||||||
};
|
};
|
||||||
use objdiff_core::{
|
use objdiff_core::{
|
||||||
arch::ObjArch,
|
arch::ObjArch,
|
||||||
|
build::BuildStatus,
|
||||||
diff::{display::HighlightKind, ObjDiff, ObjSymbolDiff},
|
diff::{display::HighlightKind, ObjDiff, ObjSymbolDiff},
|
||||||
|
jobs::{create_scratch::CreateScratchResult, objdiff::ObjDiffResult, Job, JobQueue, JobResult},
|
||||||
obj::{
|
obj::{
|
||||||
ObjInfo, ObjSection, ObjSectionKind, ObjSymbol, ObjSymbolFlags, SymbolRef, SECTION_COMMON,
|
ObjInfo, ObjSection, ObjSectionKind, ObjSymbol, ObjSymbolFlags, SymbolRef, SECTION_COMMON,
|
||||||
},
|
},
|
||||||
|
@ -16,11 +18,7 @@ use regex::{Regex, RegexBuilder};
|
||||||
use crate::{
|
use crate::{
|
||||||
app::AppStateRef,
|
app::AppStateRef,
|
||||||
hotkeys,
|
hotkeys,
|
||||||
jobs::{
|
jobs::{is_create_scratch_available, start_create_scratch},
|
||||||
create_scratch::{start_create_scratch, CreateScratchConfig, CreateScratchResult},
|
|
||||||
objdiff::{BuildStatus, ObjDiffResult},
|
|
||||||
Job, JobQueue, JobResult,
|
|
||||||
},
|
|
||||||
views::{
|
views::{
|
||||||
appearance::Appearance,
|
appearance::Appearance,
|
||||||
column_layout::{render_header, render_strips},
|
column_layout::{render_header, render_strips},
|
||||||
|
@ -182,7 +180,7 @@ impl DiffViewState {
|
||||||
} else {
|
} else {
|
||||||
self.source_path_available = false;
|
self.source_path_available = false;
|
||||||
}
|
}
|
||||||
self.scratch_available = CreateScratchConfig::is_available(&state.config);
|
self.scratch_available = is_create_scratch_available(&state.config);
|
||||||
self.object_name =
|
self.object_name =
|
||||||
state.config.selected_obj.as_ref().map(|o| o.name.clone()).unwrap_or_default();
|
state.config.selected_obj.as_ref().map(|o| o.name.clone()).unwrap_or_default();
|
||||||
}
|
}
|
||||||
|
@ -270,14 +268,7 @@ impl DiffViewState {
|
||||||
let Ok(state) = state.read() else {
|
let Ok(state) = state.read() else {
|
||||||
return;
|
return;
|
||||||
};
|
};
|
||||||
match CreateScratchConfig::from_config(&state.config, function_name) {
|
start_create_scratch(ctx, jobs, &state, function_name);
|
||||||
Ok(config) => {
|
|
||||||
jobs.push_once(Job::CreateScratch, || start_create_scratch(ctx, config));
|
|
||||||
}
|
|
||||||
Err(err) => {
|
|
||||||
log::error!("Failed to create scratch config: {err}");
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
DiffViewAction::OpenSourcePath => {
|
DiffViewAction::OpenSourcePath => {
|
||||||
let Ok(state) = state.read() else {
|
let Ok(state) = state.read() else {
|
||||||
|
|
Loading…
Reference in New Issue