Skip to content
Snippets Groups Projects
elf.rs 4.91 KiB
Newer Older
//! ELF executables

use collections::String;
#[cfg(target_arch = "x86")]
use goblin::elf32::{header, program_header};

#[cfg(target_arch = "x86_64")]
use goblin::elf64::{header, program_header};
use arch;
Jeremy Soller's avatar
Jeremy Soller committed
use arch::externs::memcpy;
use arch::paging::{entry, VirtualAddress};
use arch::start::usermode;
use context;
use syscall::{Error, Result as SysResult};
/// An ELF executable
pub struct Elf<'a> {
    pub data: &'a [u8],
    header: &'a header::Header
}

impl<'a> Elf<'a> {
    /// Create a ELF executable from data
    pub fn from(data: &'a [u8]) -> Result<Elf<'a>, String> {
        if data.len() < header::SIZEOF_EHDR {
            Err(format!("Elf: Not enough data: {} < {}", data.len(), header::SIZEOF_EHDR))
        } else if &data[..header::SELFMAG] != header::ELFMAG {
            Err(format!("Elf: Invalid magic: {:?} != {:?}", &data[..4], header::ELFMAG))
        } else if data.get(header::EI_CLASS) != Some(&header::ELFCLASS) {
            Err(format!("Elf: Invalid architecture: {:?} != {:?}", data.get(header::EI_CLASS), header::ELFCLASS))
            Ok(Elf {
                data: data,
                header: unsafe { &*(data.as_ptr() as usize as *const header::Header) }
            })
        }
    }

    pub fn segments(&'a self) -> ElfSegments<'a> {
        ElfSegments {
            data: self.data,
            header: self.header,
            i: 0
    /// Get the entry field of the header
    pub fn entry(&self) -> usize {
        self.header.e_entry as usize
    }

    /// Test function to run. Remove and replace with proper syscall
    pub fn run(self) -> SysResult<!> {
        {
            let contexts = context::contexts();
            let context_lock = contexts.current().ok_or(Error::NoProcess)?;
            let mut context = context_lock.write();

            // Unmap previous image and stack
            context.image.clear();
Jeremy Soller's avatar
Jeremy Soller committed
            drop(context.heap.take());
            drop(context.stack.take());

            for segment in self.segments() {
                if segment.p_type == program_header::PT_LOAD {
                    let mut memory = context::memory::Memory::new(
                        VirtualAddress::new(segment.p_vaddr as usize),
                        segment.p_memsz as usize,
Jeremy Soller's avatar
Jeremy Soller committed
                        entry::NO_EXECUTE | entry::WRITABLE,
                        true,
                        true
Jeremy Soller's avatar
Jeremy Soller committed
                    );

                    unsafe {
                        // Copy file data
                        memcpy(segment.p_vaddr as *mut u8,
                                (self.data.as_ptr() as usize + segment.p_offset as usize) as *const u8,
                                segment.p_filesz as usize);
                    let mut flags = entry::NO_EXECUTE | entry::USER_ACCESSIBLE;
                    if segment.p_flags & program_header::PF_R == program_header::PF_R {
                        flags.insert(entry::PRESENT);
                    }
                    // W ^ X. If it is executable, do not allow it to be writable, even if requested
                    if segment.p_flags & program_header::PF_X == program_header::PF_X {
                        flags.remove(entry::NO_EXECUTE);
                    } else if segment.p_flags & program_header::PF_W == program_header::PF_W {
                        flags.insert(entry::WRITABLE);
                    }

                    memory.remap(flags, true);
                    context.image.push(memory.to_shared());
            context.heap = Some(context::memory::Memory::new(
                VirtualAddress::new(arch::USER_HEAP_OFFSET),
                0,
                entry::NO_EXECUTE | entry::WRITABLE | entry::USER_ACCESSIBLE,
                true,
                true
            ).to_shared());

            // Map stack
            context.stack = Some(context::memory::Memory::new(
Jeremy Soller's avatar
Jeremy Soller committed
                VirtualAddress::new(arch::USER_STACK_OFFSET),
                arch::USER_STACK_SIZE,
Jeremy Soller's avatar
Jeremy Soller committed
                entry::NO_EXECUTE | entry::WRITABLE | entry::USER_ACCESSIBLE,
                true,
                true
Jeremy Soller's avatar
Jeremy Soller committed
            ));

        // Go to usermode
Jeremy Soller's avatar
Jeremy Soller committed
        unsafe { usermode(self.entry(), arch::USER_STACK_OFFSET + arch::USER_STACK_SIZE - 256); }
pub struct ElfSegments<'a> {
    data: &'a [u8],
    header: &'a header::Header,
    i: usize
}

impl<'a> Iterator for ElfSegments<'a> {
    type Item = &'a program_header::ProgramHeader;
    fn next(&mut self) -> Option<Self::Item> {
        if self.i < self.header.e_phnum as usize {
            let item = unsafe {
                &* ((
                        self.data.as_ptr() as usize
                        + self.header.e_phoff as usize
                        + self.i * self.header.e_phentsize as usize
                    ) as *const program_header::ProgramHeader)
            };
            self.i += 1;
            Some(item)
        } else {
            None
        }