use crate::call::CallFrame;
use crate::consts::*;
use crate::interpreter::{InitialBalances, Interpreter};
use derivative::Derivative;
use fuel_tx::ScriptExecutionResult;
use fuel_types::{fmt_truncated_hex, ContractId, Word};
#[derive(Derivative)]
#[derivative(Debug)]
pub struct Backtrace {
call_stack: Vec<CallFrame>,
contract: ContractId,
registers: [Word; VM_REGISTER_COUNT],
#[derivative(Debug(format_with = "fmt_truncated_hex::<16>"))]
memory: Vec<u8>,
result: ScriptExecutionResult,
initial_balances: InitialBalances,
}
impl Backtrace {
pub fn from_vm_error<S, Tx>(vm: &Interpreter<S, Tx>, result: ScriptExecutionResult) -> Self {
let call_stack = vm.call_stack().to_owned();
let contract = vm.internal_contract_or_default();
let memory = vm.memory().to_owned();
let initial_balances = vm.initial_balances().clone();
let mut registers = [0; VM_REGISTER_COUNT];
registers.copy_from_slice(vm.registers());
Self {
call_stack,
contract,
registers,
memory,
result,
initial_balances,
}
}
pub fn call_stack(&self) -> &[CallFrame] {
self.call_stack.as_slice()
}
pub const fn contract(&self) -> &ContractId {
&self.contract
}
pub const fn registers(&self) -> &[Word] {
&self.registers
}
pub fn memory(&self) -> &[u8] {
self.memory.as_slice()
}
pub const fn result(&self) -> &ScriptExecutionResult {
&self.result
}
pub const fn initial_balances(&self) -> &InitialBalances {
&self.initial_balances
}
pub fn into_inner(
self,
) -> (
Vec<CallFrame>,
ContractId,
[Word; VM_REGISTER_COUNT],
Vec<u8>,
ScriptExecutionResult,
InitialBalances,
) {
let Self {
call_stack,
contract,
registers,
memory,
result,
initial_balances,
} = self;
(call_stack, contract, registers, memory, result, initial_balances)
}
}