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