libmwemu 0.24.5

x86 32/64bits and system internals emulator, for securely emulating malware and other stuff.
Documentation
use crate::emu;
use crate::serialization;
use crate::winapi::winapi32::kernel32;

pub fn gateway(addr: u32, emu: &mut emu::Emu) -> String {
    let api = kernel32::guess_api_name(emu, addr);
    let api = api.split("!").last().unwrap_or(&api);
    match api {
        "ShellExecuteA" => ShellExecuteA(emu),
        "ShellExecuteW" => ShellExecuteW(emu),
        _ => {
            if emu.cfg.skip_unimplemented == false {
                if emu.cfg.dump_on_exit && emu.cfg.dump_filename.is_some() {
                    serialization::Serialization::dump(
                        &emu,
                        emu.cfg.dump_filename.as_ref().unwrap(),
                    );
                }

                unimplemented!("atemmpt to call unimplemented API 0x{:x} {}", addr, api);
            }
            log::warn!(
                "calling unimplemented API 0x{:x} {} at 0x{:x}",
                addr,
                api,
                emu.regs().rip
            );
            return api.to_ascii_lowercase();
        }
    }

    String::new()
}

fn ShellExecuteA(emu: &mut emu::Emu) {
    let hwnd = emu
        .maps
        .read_dword(emu.regs().get_esp())
        .expect("shell32!ShellExecuteA error reading hwnd") as u64;
    let lp_operation = emu
        .maps
        .read_dword(emu.regs().get_esp() + 4)
        .expect("shell32!ShellExecuteA error reading lp_operation") as u64;
    let lp_file = emu
        .maps
        .read_dword(emu.regs().get_esp() + 8)
        .expect("shell32!ShellExecuteA error reading lp_file") as u64;
    let lp_parameters = emu
        .maps
        .read_dword(emu.regs().get_esp() + 12)
        .expect("shell32!ShellExecuteA error reading lp_parameters") as u64;
    let lp_directory = emu
        .maps
        .read_dword(emu.regs().get_esp() + 16)
        .expect("shell32!ShellExecuteA error reading lp_directory") as u64;
    let n_show_cmd = emu
        .maps
        .read_dword(emu.regs().get_esp() + 20)
        .expect("shell32!ShellExecuteA error reading n_show_cmd") as u64;

    let operation = if lp_operation != 0 {
        emu.maps.read_string(lp_operation)
    } else {
        "open".to_string()
    };
    let file = emu.maps.read_string(lp_file);
    let params = if lp_parameters != 0 {
        emu.maps.read_string(lp_parameters)
    } else {
        "".to_string()
    };

    log_red!(
        emu,
        "shell32!ShellExecuteA op: {} file: {} params: {}",
        operation,
        file,
        params
    );

    for _ in 0..6 {
        emu.stack_pop32(false);
    }

    emu.regs_mut().rax = 42; // > 32 means success
}

fn ShellExecuteW(emu: &mut emu::Emu) {
    let hwnd = emu
        .maps
        .read_dword(emu.regs().get_esp())
        .expect("shell32!ShellExecuteW error reading hwnd") as u64;
    let lp_operation = emu
        .maps
        .read_dword(emu.regs().get_esp() + 4)
        .expect("shell32!ShellExecuteW error reading lp_operation") as u64;
    let lp_file = emu
        .maps
        .read_dword(emu.regs().get_esp() + 8)
        .expect("shell32!ShellExecuteW error reading lp_file") as u64;
    let lp_parameters = emu
        .maps
        .read_dword(emu.regs().get_esp() + 12)
        .expect("shell32!ShellExecuteW error reading lp_parameters") as u64;
    let lp_directory = emu
        .maps
        .read_dword(emu.regs().get_esp() + 16)
        .expect("shell32!ShellExecuteW error reading lp_directory") as u64;
    let n_show_cmd = emu
        .maps
        .read_dword(emu.regs().get_esp() + 20)
        .expect("shell32!ShellExecuteW error reading n_show_cmd") as u64;

    let operation = if lp_operation != 0 {
        emu.maps.read_wide_string(lp_operation)
    } else {
        "open".to_string()
    };
    let file = emu.maps.read_wide_string(lp_file);
    let params = if lp_parameters != 0 {
        emu.maps.read_wide_string(lp_parameters)
    } else {
        "".to_string()
    };

    log_red!(
        emu,
        "shell32!ShellExecuteW op: {} file: {} params: {}",
        operation,
        file,
        params
    );

    for _ in 0..6 {
        emu.stack_pop32(false);
    }

    emu.regs_mut().rax = 42; // > 32 means success
}