use crate::FaitheError;
use std::mem::size_of;
use windows::Win32::{
Foundation::{HANDLE, HINSTANCE},
System::Diagnostics::ToolHelp::{
CreateToolhelp32Snapshot, Module32FirstW, Module32NextW, MODULEENTRY32W, TH32CS_SNAPMODULE,
TH32CS_SNAPMODULE32,
},
};
#[derive(Debug, Clone)]
pub struct ModuleEntry {
pub process_id: u32,
pub mod_base_addr: usize,
pub mod_base_size: usize,
pub h_module: HINSTANCE,
pub sz_module: String,
pub sz_exe_path: String,
}
impl From<MODULEENTRY32W> for ModuleEntry {
fn from(me: MODULEENTRY32W) -> Self {
Self {
process_id: me.th32ProcessID,
mod_base_addr: me.modBaseAddr as _,
mod_base_size: me.modBaseSize as _,
h_module: me.hModule,
sz_module: String::from_utf16_lossy(
&me.szModule[..me.szModule.iter().position(|b| *b == 0).unwrap_or(0)],
),
sz_exe_path: String::from_utf16_lossy(
&me.szExePath[..me.szExePath.iter().position(|b| *b == 0).unwrap_or(0)],
),
}
}
}
pub struct Modules {
snap: HANDLE,
entry: MODULEENTRY32W,
ret: bool,
}
impl Modules {
pub fn new(process_id: u32) -> crate::Result<Self> {
unsafe {
let snap =
CreateToolhelp32Snapshot(TH32CS_SNAPMODULE | TH32CS_SNAPMODULE32, process_id).map_err(|_| FaitheError::last_error())?;
let entry = MODULEENTRY32W {
dwSize: size_of::<MODULEENTRY32W>() as _,
..Default::default()
};
let mut this = Self {
snap,
entry,
ret: true,
};
if Module32FirstW(snap, &mut this.entry) == false {
Err(FaitheError::last_error())
} else {
Ok(this)
}
}
}
}
impl Iterator for Modules {
type Item = ModuleEntry;
fn next(&mut self) -> Option<Self::Item> {
if !self.ret {
None
} else {
let this = self.entry.into();
unsafe {
self.ret = Module32NextW(self.snap, &mut self.entry) == true;
}
Some(this)
}
}
}