#![allow(clippy::arithmetic_side_effects)]
use crate::{
ebpf,
elf::Executable,
error::{EbpfError, ProgramResult},
interpreter::Interpreter,
memory_region::MemoryMapping,
program::{BuiltinFunction, BuiltinProgram, FunctionRegistry, SBPFVersion},
static_analysis::{Analysis, DummyContextObject, RegisterTraceEntry},
};
use std::{collections::BTreeMap, fmt::Debug, mem::offset_of};
#[cfg(feature = "shuttle-test")]
use shuttle::sync::Arc;
#[cfg(not(feature = "shuttle-test"))]
use std::sync::Arc;
#[cfg(all(feature = "jit", not(feature = "shuttle-test")))]
use rand::{thread_rng, Rng};
#[cfg(all(feature = "jit", feature = "shuttle-test"))]
use shuttle::rand::{thread_rng, Rng};
#[cfg(feature = "jit")]
const PROGRAM_ENVIRONMENT_KEY_SHIFT: u32 = 4;
#[cfg(feature = "jit")]
static RUNTIME_ENVIRONMENT_KEY: std::sync::OnceLock<i32> = std::sync::OnceLock::<i32>::new();
pub fn get_runtime_environment_key() -> i32 {
#[cfg(feature = "jit")]
{
*RUNTIME_ENVIRONMENT_KEY
.get_or_init(|| thread_rng().gen::<i32>() >> PROGRAM_ENVIRONMENT_KEY_SHIFT)
}
#[cfg(not(feature = "jit"))]
0
}
#[derive(Debug, Clone, PartialEq, Eq)]
pub struct Config {
pub max_call_depth: usize,
pub stack_frame_size: usize,
pub enable_address_translation: bool,
pub enable_stack_frame_gaps: bool,
pub instruction_meter_checkpoint_distance: usize,
pub enable_instruction_meter: bool,
pub enable_register_tracing: bool,
pub enable_symbol_and_section_labels: bool,
pub reject_broken_elfs: bool,
#[cfg(feature = "jit")]
pub noop_instruction_rate: u32,
#[cfg(feature = "jit")]
pub sanitize_user_provided_values: bool,
pub optimize_rodata: bool,
pub allow_memory_region_zero: bool,
pub aligned_memory_mapping: bool,
pub enabled_sbpf_versions: std::ops::RangeInclusive<SBPFVersion>,
}
impl Config {
pub fn stack_size(&self) -> usize {
self.stack_frame_size * self.max_call_depth
}
}
impl Default for Config {
fn default() -> Self {
Self {
max_call_depth: 64,
stack_frame_size: 4_096,
enable_address_translation: true,
enable_stack_frame_gaps: true,
instruction_meter_checkpoint_distance: 10000,
enable_instruction_meter: true,
enable_register_tracing: false,
enable_symbol_and_section_labels: false,
reject_broken_elfs: false,
#[cfg(feature = "jit")]
noop_instruction_rate: 256,
#[cfg(feature = "jit")]
sanitize_user_provided_values: true,
optimize_rodata: true,
allow_memory_region_zero: true,
aligned_memory_mapping: false,
enabled_sbpf_versions: SBPFVersion::V0..=SBPFVersion::V4,
}
}
}
impl<C: ContextObject> Executable<C> {
pub fn from_elf(elf_bytes: &[u8], loader: Arc<BuiltinProgram<C>>) -> Result<Self, EbpfError> {
let executable = Executable::load(elf_bytes, loader)?;
Ok(executable)
}
pub fn from_text_bytes(
text_bytes: &[u8],
loader: Arc<BuiltinProgram<C>>,
sbpf_version: SBPFVersion,
function_registry: FunctionRegistry<usize>,
) -> Result<Self, EbpfError> {
Executable::new_from_text_bytes(text_bytes, loader, sbpf_version, function_registry)
.map_err(EbpfError::ElfError)
}
}
pub trait ContextObject {
fn consume(&mut self, amount: u64);
fn get_remaining(&self) -> u64;
}
pub struct DynamicAnalysis {
pub edge_counter_max: usize,
pub edges: BTreeMap<usize, BTreeMap<usize, usize>>,
}
impl DynamicAnalysis {
pub fn new(register_trace: &[[u64; 12]], analysis: &Analysis) -> Self {
let mut result = Self {
edge_counter_max: 0,
edges: BTreeMap::new(),
};
let mut last_basic_block = usize::MAX;
for traced_instruction in register_trace.iter() {
let pc = traced_instruction[11] as usize;
if analysis.cfg_nodes.contains_key(&pc) {
let counter = result
.edges
.entry(last_basic_block)
.or_default()
.entry(pc)
.or_insert(0);
*counter += 1;
result.edge_counter_max = result.edge_counter_max.max(*counter);
last_basic_block = pc;
}
}
result
}
}
#[derive(Clone, Default)]
pub struct CallFrame {
pub caller_saved_registers: [u64; ebpf::SCRATCH_REGS],
pub frame_pointer: u64,
pub target_pc: u64,
}
pub enum RuntimeEnvironmentSlot {
HostStackPointer = offset_of!(EbpfVm<DummyContextObject>, host_stack_pointer) as isize,
CallDepth = offset_of!(EbpfVm<DummyContextObject>, call_depth) as isize,
ContextObjectPointer = offset_of!(EbpfVm<DummyContextObject>, context_object_pointer) as isize,
PreviousInstructionMeter =
offset_of!(EbpfVm<DummyContextObject>, previous_instruction_meter) as isize,
DueInsnCount = offset_of!(EbpfVm<DummyContextObject>, due_insn_count) as isize,
StopwatchNumerator = offset_of!(EbpfVm<DummyContextObject>, stopwatch_numerator) as isize,
StopwatchDenominator = offset_of!(EbpfVm<DummyContextObject>, stopwatch_denominator) as isize,
Registers = offset_of!(EbpfVm<DummyContextObject>, registers) as isize,
ProgramResult = offset_of!(EbpfVm<DummyContextObject>, program_result) as isize,
MemoryMapping = offset_of!(EbpfVm<DummyContextObject>, memory_mapping) as isize,
RegisterTrace = offset_of!(EbpfVm<DummyContextObject>, register_trace) as isize,
}
#[repr(C)]
pub struct EbpfVm<'a, C: ContextObject> {
pub host_stack_pointer: *mut u64,
pub call_depth: u64,
pub context_object_pointer: &'a mut C,
pub previous_instruction_meter: u64,
pub due_insn_count: u64,
pub stopwatch_numerator: u64,
pub stopwatch_denominator: u64,
pub registers: [u64; 12],
pub program_result: ProgramResult,
pub memory_mapping: MemoryMapping,
pub call_frames: Vec<CallFrame>,
pub loader: Arc<BuiltinProgram<C>>,
pub register_trace: Vec<RegisterTraceEntry>,
#[cfg(feature = "debugger")]
pub debug_port: Option<u16>,
}
impl<'a, C: ContextObject> EbpfVm<'a, C> {
pub fn new(
loader: Arc<BuiltinProgram<C>>,
sbpf_version: SBPFVersion,
context_object: &'a mut C,
mut memory_mapping: MemoryMapping,
stack_len: usize,
) -> Self {
let config = loader.get_config();
let mut registers = [0u64; 12];
registers[ebpf::FRAME_PTR_REG] =
ebpf::MM_STACK_START.saturating_add(if !sbpf_version.manual_stack_frame_bump() {
config.stack_frame_size
} else {
stack_len
} as u64);
if !config.enable_address_translation {
memory_mapping = MemoryMapping::new_identity();
}
EbpfVm {
host_stack_pointer: std::ptr::null_mut(),
call_depth: 0,
context_object_pointer: context_object,
previous_instruction_meter: 0,
due_insn_count: 0,
stopwatch_numerator: 0,
stopwatch_denominator: 0,
registers,
program_result: ProgramResult::Ok(0),
memory_mapping,
call_frames: vec![CallFrame::default(); config.max_call_depth],
loader,
#[cfg(feature = "debugger")]
debug_port: std::env::var("VM_DEBUG_PORT")
.ok()
.and_then(|v| v.parse::<u16>().ok()),
register_trace: Vec::default(),
}
}
pub fn execute_program(
&mut self,
executable: &Executable<C>,
interpreted: bool,
) -> (u64, ProgramResult) {
debug_assert!(Arc::ptr_eq(&self.loader, executable.get_loader()));
self.registers[11] = executable.get_entrypoint_instruction_offset() as u64;
let config = executable.get_config();
let initial_insn_count = self.context_object_pointer.get_remaining();
self.previous_instruction_meter = initial_insn_count;
self.due_insn_count = 0;
self.program_result = ProgramResult::Ok(0);
if interpreted {
#[cold]
#[inline(never)]
#[cfg(feature = "debugger")]
fn run_interpreter<C: ContextObject>(mut interpreter: Interpreter<C>) {
let debug_port = interpreter.vm.debug_port.clone();
if let Some(debug_port) = debug_port {
crate::debugger::execute(&mut interpreter, debug_port);
} else {
while interpreter.step() {}
}
}
#[cold]
#[inline(never)]
#[cfg(not(feature = "debugger"))]
fn run_interpreter<C: ContextObject>(mut interpreter: Interpreter<C>) {
while interpreter.step() {}
}
let interpreter = Interpreter::new(self, executable, self.registers);
run_interpreter(interpreter);
} else {
#[cfg(all(feature = "jit", not(target_os = "windows"), target_arch = "x86_64"))]
{
let compiled_program = match executable
.get_compiled_program()
.ok_or_else(|| EbpfError::JitNotCompiled)
{
Ok(compiled_program) => compiled_program,
Err(error) => return (0, ProgramResult::Err(error)),
};
compiled_program.invoke(config, self, self.registers);
}
#[cfg(not(all(feature = "jit", not(target_os = "windows"), target_arch = "x86_64")))]
{
return (0, ProgramResult::Err(EbpfError::JitNotCompiled));
}
};
let instruction_count = if config.enable_instruction_meter {
self.context_object_pointer.consume(self.due_insn_count);
initial_insn_count.saturating_sub(self.context_object_pointer.get_remaining())
} else {
0
};
let mut result = ProgramResult::Ok(0);
std::mem::swap(&mut result, &mut self.program_result);
(instruction_count, result)
}
pub fn invoke_function(&mut self, function: BuiltinFunction<C>) {
function(
unsafe {
std::ptr::addr_of_mut!(*self)
.cast::<u64>()
.offset(get_runtime_environment_key() as isize)
.cast::<Self>()
},
self.registers[1],
self.registers[2],
self.registers[3],
self.registers[4],
self.registers[5],
);
}
}