use std::env;
use std::fs;
use std::io::{self, Write};
use std::process::Command;
use crate::error::LedgerError;
#[derive(Default, Debug)]
pub struct LedgerAppInfos {
pub api_level: String,
pub target_id: Option<String>,
pub size: u64,
}
fn get_string_from_offset(
vector: &[u8],
offset: &usize,
) -> Result<String, LedgerError> {
let end_index = vector[*offset..]
.iter()
.position(|&x| x == b'\n')
.map(|pos| *offset + pos)
.unwrap_or(*offset); String::from_utf8(vector[*offset..end_index].to_vec())
.map_err(|e| LedgerError::Other(format!("Invalid UTF-8: {e}")))
}
pub fn retrieve_infos(
file: &std::path::Path,
) -> Result<LedgerAppInfos, LedgerError> {
let buffer = fs::read(file)?;
let elf = goblin::elf::Elf::parse(&buffer)?;
let mut infos = LedgerAppInfos::default();
for section in elf.section_headers.iter() {
if let Some(Ok(name)) = elf.shdr_strtab.get(section.sh_name) {
if name == "ledger.api_level" {
infos.api_level = get_string_from_offset(
&buffer,
&(section.sh_offset as usize),
)?;
} else if name == ".ledger.api_level" {
infos.api_level =
buffer[section.sh_offset as usize].to_string();
} else if name == "ledger.target_id" {
infos.target_id = Some(get_string_from_offset(
&buffer,
&(section.sh_offset as usize),
)?);
}
}
}
let mut nvram_data = 0;
let mut envram_data = 0;
for s in elf.syms.iter() {
let symbol_name = elf
.strtab
.get(s.st_name)
.ok_or_else(|| LedgerError::Other("Missing symbol name".into()))?;
let name = symbol_name
.map_err(|e| LedgerError::Other(format!("Strtab error: {e}")))?;
match name {
"_nvram_data" => nvram_data = s.st_value,
"_envram_data" => envram_data = s.st_value,
_ => (),
}
}
infos.size = envram_data - nvram_data;
Ok(infos)
}
pub fn export_binary(
elf_path: &std::path::Path,
dest_bin: &std::path::Path,
) -> Result<(), LedgerError> {
let objcopy = env::var_os("CARGO_TARGET_THUMBV6M_NONE_EABI_OBJCOPY")
.unwrap_or_else(|| "arm-none-eabi-objcopy".into());
let copy_out = Command::new(&objcopy)
.arg(elf_path)
.arg(dest_bin)
.args(["-O", "ihex"])
.output()?;
if !copy_out.status.success() {
return Err(LedgerError::CommandFailure {
cmd: "objcopy",
status: copy_out.status.code(),
stderr: String::from_utf8_lossy(©_out.stderr).into(),
});
}
let size = env::var_os("CARGO_TARGET_THUMBV6M_NONE_EABI_SIZE")
.unwrap_or_else(|| "arm-none-eabi-size".into());
let out = Command::new(&size).arg(elf_path).output()?;
if !out.status.success() {
return Err(LedgerError::CommandFailure {
cmd: "size",
status: out.status.code(),
stderr: String::from_utf8_lossy(&out.stderr).into(),
});
}
io::stdout().write_all(&out.stdout)?;
io::stderr().write_all(&out.stderr)?;
Ok(())
}
pub fn install_with_ledgerctl(
dir: &std::path::Path,
app_json: &std::path::Path,
) -> Result<(), LedgerError> {
let out = Command::new("ledgerctl")
.current_dir(dir)
.args(["install", "-f", app_json.to_str().unwrap()])
.output()?;
if !out.status.success() {
return Err(LedgerError::CommandFailure {
cmd: "ledgerctl",
status: out.status.code(),
stderr: String::from_utf8_lossy(&out.stderr).into(),
});
}
io::stdout().write_all(&out.stdout)?;
io::stderr().write_all(&out.stderr)?;
Ok(())
}
pub fn dump_with_ledgerctl(
dir: &std::path::Path,
app_json: &std::path::Path,
out_file_name: &str,
) -> Result<(), LedgerError> {
let out = Command::new("ledgerctl")
.current_dir(dir)
.args([
"install",
app_json.to_str().unwrap(),
"-o",
out_file_name,
"-f",
])
.output()?;
if !out.status.success() {
return Err(LedgerError::CommandFailure {
cmd: "ledgerctl",
status: out.status.code(),
stderr: String::from_utf8_lossy(&out.stderr).into(),
});
}
io::stdout().write_all(&out.stdout)?;
io::stderr().write_all(&out.stderr)?;
Ok(())
}