1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
use super::OwnedProcess;
use crate::{module::ModuleIterator, thread::ThreadIterator, FaitheError};
use std::mem::size_of;
use windows::Win32::{
Foundation::HANDLE,
System::{
Diagnostics::ToolHelp::{
CreateToolhelp32Snapshot, Process32FirstW, Process32NextW, PROCESSENTRY32W,
TH32CS_SNAPPROCESS,
},
Threading::PROCESS_ACCESS_RIGHTS,
},
};
#[derive(Debug, Clone)]
pub struct ProcessEntry {
pub process_id: u32,
pub cnt_threads: u32,
pub parent_id: u32,
pub thread_base_priority: i32,
pub file_name: String,
}
impl ProcessEntry {
pub fn modules(&self) -> crate::Result<ModuleIterator> {
ModuleIterator::new(self.process_id)
}
pub fn threads(&self) -> crate::Result<ThreadIterator> {
ThreadIterator::new(self.process_id)
}
}
impl ProcessEntry {
pub fn open(
&self,
inherit_handle: bool,
desired_access: PROCESS_ACCESS_RIGHTS,
) -> crate::Result<OwnedProcess> {
OwnedProcess::open_by_id(self.process_id, inherit_handle, desired_access)
}
}
impl From<PROCESSENTRY32W> for ProcessEntry {
fn from(pe: PROCESSENTRY32W) -> Self {
Self {
process_id: pe.th32ProcessID,
cnt_threads: pe.cntThreads,
parent_id: pe.th32ParentProcessID,
thread_base_priority: pe.pcPriClassBase,
file_name: String::from_utf16_lossy(
&pe.szExeFile[..pe.szExeFile.iter().position(|v| *v == 0).unwrap_or(0)],
),
}
}
}
pub struct ProcessIterator {
snap: HANDLE,
entry: PROCESSENTRY32W,
ret: bool,
}
impl ProcessIterator {
pub fn new() -> crate::Result<Self> {
unsafe {
let snap = CreateToolhelp32Snapshot(TH32CS_SNAPPROCESS, 0)
.map_err(|_| FaitheError::last_error())?;
let entry = PROCESSENTRY32W {
dwSize: size_of::<PROCESSENTRY32W>() as _,
..Default::default()
};
let mut this = Self {
snap,
entry,
ret: true,
};
if Process32FirstW(this.snap, &mut this.entry) == false {
Err(FaitheError::last_error())
} else {
Ok(this)
}
}
}
}
impl Iterator for ProcessIterator {
type Item = ProcessEntry;
fn next(&mut self) -> Option<Self::Item> {
if !self.ret {
None
} else {
let this = self.entry.into();
unsafe {
self.ret = Process32NextW(self.snap, &mut self.entry) == true;
}
Some(this)
}
}
}