Skip to content

Commit

Permalink
Unify BPF verifiers
Browse files Browse the repository at this point in the history
  • Loading branch information
jackcmay committed May 27, 2021
1 parent da47fd0 commit 38927bd
Show file tree
Hide file tree
Showing 7 changed files with 99 additions and 194 deletions.
2 changes: 1 addition & 1 deletion src/assembler.rs
Original file line number Diff line number Diff line change
Expand Up @@ -212,7 +212,7 @@ fn insn(opc: u8, dst: i64, src: i64, off: i64, imm: i64) -> Result<Insn, String>
/// ```
pub fn assemble<E: UserDefinedError, I: 'static + InstructionMeter>(
src: &str,
verifier: Option<Verifier<E>>,
verifier: Option<Verifier>,
config: Config,
) -> Result<Box<dyn Executable<E, I>>, String> {
fn resolve_label(
Expand Down
2 changes: 0 additions & 2 deletions src/ebpf.rs
Original file line number Diff line number Diff line change
Expand Up @@ -25,8 +25,6 @@ use std::fmt;
pub const PROG_MAX_INSNS: usize = 65_536;
/// Size of an eBPF instructions, in bytes.
pub const INSN_SIZE: usize = 8;
/// Maximum size of an eBPF program, in bytes.
pub const PROG_MAX_SIZE: usize = PROG_MAX_INSNS * INSN_SIZE;
/// Stack register
pub const STACK_REG: usize = 10;
/// First scratch register
Expand Down
5 changes: 4 additions & 1 deletion src/error.rs
Original file line number Diff line number Diff line change
Expand Up @@ -17,7 +17,7 @@
//! <https://www.kernel.org/doc/Documentation/networking/filter.txt>, or for a shorter version of
//! the list of the operation codes: <https://github.com/iovisor/bpf-docs/blob/master/eBPF.md>
use crate::{elf::ElfError, memory_region::AccessType};
use crate::{elf::ElfError, memory_region::AccessType, verifier::VerifierError};

/// User defined errors must implement this trait
pub trait UserDefinedError: 'static + std::error::Error {}
Expand Down Expand Up @@ -86,4 +86,7 @@ pub enum EbpfError<E: UserDefinedError> {
/// Compilation is too big to fit
#[error("Compilation exhaused text segment at instruction {0}")]
ExhausedTextSegment(usize),
/// ELF error
#[error("Verifier error: {0}")]
VerifierError(#[from] VerifierError),
}
98 changes: 46 additions & 52 deletions src/verifier.rs
Original file line number Diff line number Diff line change
Expand Up @@ -21,44 +21,44 @@
//!
//! Contrary to the verifier of the Linux kernel, this one does not modify the bytecode at all.
use crate::{ebpf, error::UserDefinedError, user_error::UserError};
#![allow(clippy::upper_case_acronyms)]

use crate::ebpf;
use thiserror::Error;

/// Error definitions
#[derive(Debug, thiserror::Error, PartialEq, Eq)]
#[derive(Debug, Error, Eq, PartialEq)]
pub enum VerifierError {
/// ProgramLengthNotMultiple
#[error("program length must be a multiple of {} octets", ebpf::INSN_SIZE)]
ProgramLengthNotMultiple,
/// ProgramTooLarge
#[error("program too big, max {}, is {}", ebpf::PROG_MAX_INSNS, .0)]
/// Deprecated
#[error("Deprecated")]
ProgramTooLarge(usize),
/// NoProgram
#[error("no program set, call prog_set() to load one")]
NoProgram,
///InvalidLastInstruction
#[error("program does not end with “EXIT” instruction")]
InvalidLastInstruction,
/// DivisionByZero
/// Division by zero
#[error("division by 0 (insn #{0})")]
DivisionByZero(usize),
/// UnsupportedLeBeArgument
/// UnsupportedLEBEArgument
#[error("unsupported argument for LE/BE (insn #{0})")]
UnsupportedLeBeArgument(usize),
/// LddwCannotBeLast
UnsupportedLEBEArgument(usize),
/// LDDWCannotBeLast
#[error("LD_DW instruction cannot be last in program")]
LddwCannotBeLast,
/// IncompleteLddw
LDDWCannotBeLast,
/// IncompleteLDDW
#[error("incomplete LD_DW instruction (insn #{0})")]
IncompleteLddw(usize),
IncompleteLDDW(usize),
/// InfiniteLoop
#[error("infinite loop (insn #{0})")]
InfiniteLoop(usize),
/// JumpOutOfCode
#[error("jump out of code to #{0} (insn #{1})")]
JumpOutOfCode(usize, usize),
/// JumpToMiddleOfLddw
/// JumpToMiddleOfLDDW
#[error("jump to middle of LD_DW at #{0} (insn #{1})")]
JumpToMiddleOfLddw(usize, usize),
JumpToMiddleOfLDDW(usize, usize),
/// InvalidSourceRegister
#[error("invalid source register (insn #{0})")]
InvalidSourceRegister(usize),
Expand All @@ -78,109 +78,104 @@ pub enum VerifierError {
#[error("Invalid register specified at instruction {0}")]
InvalidRegister(usize),
}
impl UserDefinedError for VerifierError {}
impl From<VerifierError> for UserError {
fn from(error: VerifierError) -> Self {
UserError::VerifierError(error)
}

fn adj_insn_ptr(insn_ptr: usize) -> usize {
insn_ptr + ebpf::ELF_INSN_DUMP_OFFSET
}

fn check_prog_len(prog: &[u8]) -> Result<(), VerifierError> {
if prog.len() % ebpf::INSN_SIZE != 0 {
return Err(VerifierError::ProgramLengthNotMultiple);
}
if prog.len() > ebpf::PROG_MAX_SIZE {
return Err(VerifierError::ProgramTooLarge(prog.len() / ebpf::INSN_SIZE));
}

if prog.is_empty() {
return Err(VerifierError::NoProgram);
}
let last_insn = ebpf::get_insn(prog, (prog.len() / ebpf::INSN_SIZE) - 1);
if last_insn.opc != ebpf::EXIT {
return Err(VerifierError::InvalidLastInstruction);
}

Ok(())
}

fn check_imm_nonzero(insn: &ebpf::Insn, insn_ptr: usize) -> Result<(), VerifierError> {
if insn.imm == 0 {
return Err(VerifierError::DivisionByZero(insn_ptr));
return Err(VerifierError::DivisionByZero(adj_insn_ptr(insn_ptr)));
}
Ok(())
}

fn check_imm_endian(insn: &ebpf::Insn, insn_ptr: usize) -> Result<(), VerifierError> {
match insn.imm {
16 | 32 | 64 => Ok(()),
_ => Err(VerifierError::UnsupportedLeBeArgument(insn_ptr)),
_ => Err(VerifierError::UnsupportedLEBEArgument(adj_insn_ptr(
insn_ptr,
))),
}
}

fn check_load_dw(prog: &[u8], insn_ptr: usize) -> Result<(), VerifierError> {
// We know we can reach next insn since we enforce an EXIT insn at the end of program, while
// this function should be called only for LD_DW insn, that cannot be last in program.
if insn_ptr + 1 >= (prog.len() / ebpf::INSN_SIZE) {
// Last instruction cannot be LD_DW because there would be no 2nd DW
return Err(VerifierError::LDDWCannotBeLast);
}
let next_insn = ebpf::get_insn(prog, insn_ptr + 1);
if next_insn.opc != 0 {
return Err(VerifierError::IncompleteLddw(insn_ptr));
return Err(VerifierError::IncompleteLDDW(adj_insn_ptr(insn_ptr)));
}
Ok(())
}

fn check_jmp_offset(prog: &[u8], insn_ptr: usize) -> Result<(), VerifierError> {
let insn = ebpf::get_insn(prog, insn_ptr);
if insn.off == -1 {
return Err(VerifierError::InfiniteLoop(insn_ptr));
}
// if insn.off == -1 {
// return Err(VerifierError::InfiniteLoop(adj_insn_ptr(insn_ptr)).into());
// }

let dst_insn_ptr = insn_ptr as isize + 1 + insn.off as isize;
if dst_insn_ptr < 0 || dst_insn_ptr as usize >= (prog.len() / ebpf::INSN_SIZE) {
return Err(VerifierError::JumpOutOfCode(
dst_insn_ptr as usize,
insn_ptr,
adj_insn_ptr(insn_ptr),
));
}
let dst_insn = ebpf::get_insn(prog, dst_insn_ptr as usize);
if dst_insn.opc == 0 {
return Err(VerifierError::JumpToMiddleOfLddw(
return Err(VerifierError::JumpToMiddleOfLDDW(
dst_insn_ptr as usize,
insn_ptr,
adj_insn_ptr(insn_ptr),
));
}
Ok(())
}

fn check_registers(insn: &ebpf::Insn, store: bool, insn_ptr: usize) -> Result<(), VerifierError> {
if insn.src > 10 {
return Err(VerifierError::InvalidSourceRegister(insn_ptr));
return Err(VerifierError::InvalidSourceRegister(adj_insn_ptr(insn_ptr)));
}
match (insn.dst, store) {
(0..=9, _) | (10, true) => Ok(()),
(10, false) => Err(VerifierError::CannotWriteR10(insn_ptr)),
(_, _) => Err(VerifierError::InvalidDestinationRegister(insn_ptr)),
(10, false) => Err(VerifierError::CannotWriteR10(adj_insn_ptr(insn_ptr))),
(_, _) => Err(VerifierError::InvalidDestinationRegister(adj_insn_ptr(
insn_ptr,
))),
}
}

/// Check that the imm is a valid shift operand
fn check_imm_shift(insn: &ebpf::Insn, insn_ptr: usize) -> Result<(), VerifierError> {
if insn.imm < 0 || insn.imm as u64 >= 64 {
return Err(VerifierError::ShiftWithOverflow(insn_ptr));
return Err(VerifierError::ShiftWithOverflow(adj_insn_ptr(insn_ptr)));
}
Ok(())
}

/// Check that the imm is a valid register number
fn check_imm_register(insn: &ebpf::Insn, insn_ptr: usize) -> Result<(), VerifierError> {
if insn.imm < 0 || insn.imm > 10 {
return Err(VerifierError::InvalidRegister(insn_ptr));
return Err(VerifierError::InvalidRegister(adj_insn_ptr(insn_ptr)));
}
Ok(())
}

/// Default eBPF verifier
/// Check the program against the verifier's rules
#[rustfmt::skip]
pub fn check(prog: &[u8]) -> Result<(), UserError> {
pub fn check(prog: &[u8]) -> Result<(), VerifierError> {
check_prog_len(prog)?;

let mut insn_ptr: usize = 0;
Expand All @@ -201,7 +196,6 @@ pub fn check(prog: &[u8]) -> Result<(), UserError> {
ebpf::LD_IND_DW => {},

ebpf::LD_DW_IMM => {
store = true;
check_load_dw(prog, insn_ptr)?;
insn_ptr += 1;
},
Expand Down Expand Up @@ -258,7 +252,7 @@ pub fn check(prog: &[u8]) -> Result<(), UserError> {
ebpf::ADD64_REG => {},
ebpf::SUB64_IMM => {},
ebpf::SUB64_REG => {},
ebpf::MUL64_IMM => {},
ebpf::MUL64_IMM => { check_imm_nonzero(&insn, insn_ptr)?; },
ebpf::MUL64_REG => {},
ebpf::DIV64_IMM => { check_imm_nonzero(&insn, insn_ptr)?; },
ebpf::DIV64_REG => {},
Expand Down Expand Up @@ -309,7 +303,7 @@ pub fn check(prog: &[u8]) -> Result<(), UserError> {
ebpf::EXIT => {},

_ => {
return Err(VerifierError::UnknownOpCode(insn.opc, insn_ptr).into());
return Err(VerifierError::UnknownOpCode(insn.opc, adj_insn_ptr(insn_ptr)));
}
}

Expand All @@ -320,7 +314,7 @@ pub fn check(prog: &[u8]) -> Result<(), UserError> {

// insn_ptr should now be equal to number of instructions.
if insn_ptr != prog.len() / ebpf::INSN_SIZE {
return Err(VerifierError::JumpOutOfCode(insn_ptr, insn_ptr).into());
return Err(VerifierError::JumpOutOfCode(adj_insn_ptr(insn_ptr), adj_insn_ptr(insn_ptr)));
}

Ok(())
Expand Down
9 changes: 5 additions & 4 deletions src/vm.rs
Original file line number Diff line number Diff line change
Expand Up @@ -20,6 +20,7 @@ use crate::{
jit::{JitProgram, JitProgramArgument},
memory_region::{AccessType, MemoryMapping, MemoryRegion},
user_error::UserError,
verifier::VerifierError,
};
use log::debug;
use std::{
Expand All @@ -36,7 +37,7 @@ use std::{
/// - Unknown instructions.
/// - Bad formed instruction.
/// - Unknown eBPF syscall index.
pub type Verifier<E> = fn(prog: &[u8]) -> Result<(), E>;
pub type Verifier = fn(prog: &[u8]) -> Result<(), VerifierError>;

/// Return value of programs and syscalls
pub type ProgramResult<E> = Result<u64, EbpfError<E>>;
Expand Down Expand Up @@ -225,7 +226,7 @@ impl<E: UserDefinedError, I: 'static + InstructionMeter> dyn Executable<E, I> {
/// Creates a post relocaiton/fixup executable from an ELF file
pub fn from_elf(
elf_bytes: &[u8],
verifier: Option<Verifier<E>>,
verifier: Option<Verifier>,
config: Config,
) -> Result<Box<Self>, EbpfError<E>> {
let ebpf_elf = EBpfElf::load(config, elf_bytes)?;
Expand All @@ -239,11 +240,11 @@ impl<E: UserDefinedError, I: 'static + InstructionMeter> dyn Executable<E, I> {
pub fn from_text_bytes(
text_bytes: &[u8],
bpf_functions: BTreeMap<u32, (usize, String)>,
verifier: Option<Verifier<E>>,
verifier: Option<Verifier>,
config: Config,
) -> Result<Box<Self>, EbpfError<E>> {
if let Some(verifier) = verifier {
verifier(text_bytes)?;
verifier(text_bytes).map_err(EbpfError::VerifierError)?;
}
Ok(Box::new(EBpfElf::new_from_text_bytes(
config,
Expand Down
Loading

0 comments on commit 38927bd

Please sign in to comment.