starnix_core/task/
tracing.rs1use crate::task::{Kernel, PidTable};
6use starnix_logging::{log_debug, log_error, log_info};
7use starnix_sync::RwLock;
8use starnix_uapi::{pid_t, tid_t};
9use std::collections::HashMap;
10use std::collections::hash_map::Entry;
11use std::sync::{Arc, Weak};
12use zx::Koid;
13
14#[derive(Debug, Clone)]
15pub struct KoidPair {
16 pub process: Option<Koid>,
17 pub thread: Option<Koid>,
18}
19
20pub type PidToKoidMap = Arc<RwLock<HashMap<tid_t, KoidPair>>>;
22
23pub struct TracePerformanceEventManager {
24 map: PidToKoidMap,
28
29 local_map: HashMap<tid_t, KoidPair>,
33
34 weak_kernel: Weak<Kernel>,
38}
39
40impl Drop for TracePerformanceEventManager {
41 fn drop(&mut self) {
42 self.stop();
44 }
45}
46
47impl TracePerformanceEventManager {
48 pub fn new() -> Self {
49 Self { map: PidToKoidMap::default(), local_map: HashMap::new(), weak_kernel: Weak::new() }
50 }
51
52 pub fn start(&mut self, kernel: &Arc<Kernel>) {
64 if self.weak_kernel.upgrade().is_some() {
68 log_error!(
69 "TracePerformanceEventManager has already been started. Re-initializing mapping"
70 );
71 }
72
73 self.weak_kernel = Arc::downgrade(kernel);
74 *kernel.pid_to_koid_mapping.write() = Some(self.map.clone());
75
76 let kernel_pids = kernel.pids.read();
77 let existing_pid_map = Self::read_existing_pid_map(&*kernel_pids);
78 self.map.write().extend(existing_pid_map);
79 }
80
81 pub fn stop(&mut self) {
84 if let Some(kernel) = self.weak_kernel.upgrade() {
85 log_info!("Stopping trace pid mapping. Notifier set to None.");
86 *kernel.pid_to_koid_mapping.write() = None;
87 self.weak_kernel = Weak::new();
88 }
89 }
90
91 pub fn clear(&mut self) {
94 self.map.write().clear();
95 self.local_map.clear();
96 }
97
98 fn get_mapping(&mut self, pid: pid_t) -> &KoidPair {
102 if self.local_map.is_empty() {
103 let shared_map = self.map.read().clone();
104 self.local_map.extend(shared_map);
105 }
106
107 match self.local_map.entry(pid) {
108 Entry::Occupied(o) => o.into_mut(),
109 Entry::Vacant(v) => {
110 let shared_map = self.map.read();
114 let koid_pair = shared_map.get(&pid).expect("all pids should have mappings");
115 v.insert(koid_pair.clone())
116 }
117 }
118 }
119
120 pub fn map_pid_to_koid(&mut self, pid: pid_t) -> Koid {
122 self.get_mapping(pid).process.expect("all pids should have a process koid.")
123 }
124
125 pub fn map_tid_to_koid(&mut self, tid: tid_t) -> Koid {
127 self.get_mapping(tid).thread.expect("all tids should have a thread koid.")
128 }
129
130 fn read_existing_pid_map(pid_table: &PidTable) -> HashMap<tid_t, KoidPair> {
132 let mut pid_map = HashMap::new();
133
134 let ids = pid_table.running_task_ids();
135 for tid in &ids {
136 let Ok(task) = pid_table.get_task(*tid) else {
139 continue;
140 };
141 let Ok(running_state) = task.running_state() else {
142 continue;
143 };
144 let pair = KoidPair {
145 process: task.thread_group().get_process_koid().ok(),
146 thread: running_state.thread.get().map(|t| t.koid),
147 };
148 if pair.process.is_some() || pair.thread.is_some() {
150 pid_map.insert(*tid, pair);
151 }
152 }
153
154 log_debug!("Initialized {} pid mappings. From {} ids", pid_map.len(), ids.len());
155 pid_map
156 }
157}
158
159#[cfg(test)]
160mod tests {
161 use super::*;
162 use crate::task::ZirconThread;
163 use crate::testing::{create_task, spawn_kernel_and_run};
164 use futures::channel::oneshot;
165
166 #[fuchsia::test]
167 async fn test_initialize_pid_map() {
168 let (sender, receiver) = oneshot::channel();
169 spawn_kernel_and_run(async move |locked, current_task| {
170 let kernel = current_task.kernel();
171 let pid = current_task.task.tid;
172 let tkoid = current_task.running_state().thread.get().map(|t| t.koid);
173 let pkoid = current_task.thread_group().get_process_koid().ok();
174
175 let _another_current = create_task(locked, &kernel, "another-task");
176
177 let pid_map = TracePerformanceEventManager::read_existing_pid_map(&*kernel.pids.read());
178
179 assert!(tkoid.is_some());
180 assert_eq!(pid_map.len(), 2, "Expected 2 entries in pid_map got {pid_map:?}");
181 assert!(pid_map.contains_key(&pid));
182
183 let pair = pid_map.get(&pid).unwrap();
184 assert_eq!(pair.process, pkoid);
185 assert_eq!(pair.thread, tkoid);
186 sender.send(()).unwrap();
187 })
188 .await;
189 receiver.await.unwrap();
190 }
191
192 #[fuchsia::test]
193 fn test_mapping() {
194 let mut manager = TracePerformanceEventManager::new();
195 let mut map = HashMap::new();
196 map.insert(
197 1,
198 KoidPair { process: Some(Koid::from_raw(101)), thread: Some(Koid::from_raw(201)) },
199 );
200 map.insert(2, KoidPair { process: Some(Koid::from_raw(102)), thread: None });
201 manager.map.write().extend(map);
202
203 assert_eq!(manager.map_pid_to_koid(1), Koid::from_raw(101));
204 assert_eq!(manager.map_tid_to_koid(1), Koid::from_raw(201));
205 assert_eq!(manager.map_pid_to_koid(2), Koid::from_raw(102));
206 }
207
208 #[fuchsia::test]
209 #[should_panic]
210 fn test_unmapped_tid() {
211 let mut manager = TracePerformanceEventManager::new();
212
213 manager.map_tid_to_koid(2);
214 }
215
216 #[fuchsia::test]
217 async fn test_lifecycle() {
218 let (sender, receiver) = oneshot::channel();
219 spawn_kernel_and_run(async move |locked, current_task| {
220 let kernel = current_task.kernel();
221 let mut manager = TracePerformanceEventManager::new();
222
223 manager.start(&kernel);
224
225 let pid_map = manager.map.read().clone();
226 assert_eq!(pid_map.len(), 1, "Expected 1 entry in pid_map got {pid_map:?}");
227
228 let another_current = create_task(locked, &kernel, "another-task");
230 let test_thread = another_current
231 .thread_group()
232 .process
233 .create_thread(b"my-new-test-thread")
234 .expect("test thread");
235
236 {
237 another_current
238 .running_state()
239 .thread
240 .set(ZirconThread::new(Arc::new(test_thread)))
241 .expect("test thread set");
242 }
243
244 let pid_map = manager.map.read().clone();
245 let pid_dump = format!("{pid_map:?}");
246 assert_eq!(pid_map.len(), 1, "Expected 1 entry in pid_map got {pid_dump}");
247
248 another_current.record_pid_koid_mapping();
250
251 let pid_map = manager.map.read().clone();
253 let pid_dump = format!("{pid_map:?}");
254 assert_eq!(pid_map.len(), 2, "Expected 2 entries in pid_map got {pid_dump}");
255
256 let _ = manager.map_pid_to_koid(another_current.task.get_pid());
258 let _ = manager.map_pid_to_koid(another_current.task.get_tid());
259
260 manager.stop();
261
262 manager.clear();
263 assert!(manager.map.read().is_empty());
264 sender.send(()).unwrap();
265 })
266 .await;
267 receiver.await.unwrap();
268 }
269}