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.task_ids();
135 for tid in &ids {
136 let task = pid_table.get_task(*tid).expect("Empty mapping for {tid}.");
137 let live = task.live().unwrap_or_else(|_| panic!("tid is not live, tid={tid}"));
139 let pair = KoidPair {
141 process: task.thread_group().get_process_koid().ok(),
142 thread: live.thread.read().koid(),
143 };
144 if pair.process.is_some() || pair.thread.is_some() {
146 pid_map.insert(*tid, pair);
147 }
148 }
149
150 log_debug!("Initialized {} pid mappings. From {} ids", pid_map.len(), ids.len());
151 pid_map
152 }
153}
154
155#[cfg(test)]
156mod tests {
157 use super::*;
158 use crate::testing::{create_task, spawn_kernel_and_run};
159 use futures::channel::oneshot;
160
161 #[fuchsia::test]
162 async fn test_initialize_pid_map() {
163 let (sender, receiver) = oneshot::channel();
164 spawn_kernel_and_run(async move |locked, current_task| {
165 let kernel = current_task.kernel();
166 let pid = current_task.task.tid;
167 let tkoid = current_task.live().thread.read().koid();
168 let pkoid = current_task.thread_group().get_process_koid().ok();
169
170 let _another_current = create_task(locked, &kernel, "another-task");
171
172 let pid_map = TracePerformanceEventManager::read_existing_pid_map(&*kernel.pids.read());
173
174 assert!(tkoid.is_some());
175 assert_eq!(pid_map.len(), 2, "Expected 2 entries in pid_map got {pid_map:?}");
176 assert!(pid_map.contains_key(&pid));
177
178 let pair = pid_map.get(&pid).unwrap();
179 assert_eq!(pair.process, pkoid);
180 assert_eq!(pair.thread, tkoid);
181 sender.send(()).unwrap();
182 })
183 .await;
184 receiver.await.unwrap();
185 }
186
187 #[fuchsia::test]
188 fn test_mapping() {
189 let mut manager = TracePerformanceEventManager::new();
190 let mut map = HashMap::new();
191 map.insert(
192 1,
193 KoidPair { process: Some(Koid::from_raw(101)), thread: Some(Koid::from_raw(201)) },
194 );
195 map.insert(2, KoidPair { process: Some(Koid::from_raw(102)), thread: None });
196 manager.map.write().extend(map);
197
198 assert_eq!(manager.map_pid_to_koid(1), Koid::from_raw(101));
199 assert_eq!(manager.map_tid_to_koid(1), Koid::from_raw(201));
200 assert_eq!(manager.map_pid_to_koid(2), Koid::from_raw(102));
201 }
202
203 #[fuchsia::test]
204 #[should_panic]
205 fn test_unmapped_tid() {
206 let mut manager = TracePerformanceEventManager::new();
207
208 manager.map_tid_to_koid(2);
209 }
210
211 #[fuchsia::test]
212 async fn test_lifecycle() {
213 let (sender, receiver) = oneshot::channel();
214 spawn_kernel_and_run(async move |locked, current_task| {
215 let kernel = current_task.kernel();
216 let mut manager = TracePerformanceEventManager::new();
217
218 manager.start(&kernel);
219
220 let pid_map = manager.map.read().clone();
221 assert_eq!(pid_map.len(), 1, "Expected 1 entry in pid_map got {pid_map:?}");
222
223 let another_current = create_task(locked, &kernel, "another-task");
225 let test_thread = another_current
226 .thread_group()
227 .process
228 .create_thread(b"my-new-test-thread")
229 .expect("test thread");
230
231 {
232 let another_current_live = another_current.live();
233 another_current_live.thread.write().set(Arc::new(test_thread));
234 }
235
236 let pid_map = manager.map.read().clone();
237 let pid_dump = format!("{pid_map:?}");
238 assert_eq!(pid_map.len(), 1, "Expected 1 entry in pid_map got {pid_dump}");
239
240 another_current.record_pid_koid_mapping();
242
243 let pid_map = manager.map.read().clone();
245 let pid_dump = format!("{pid_map:?}");
246 assert_eq!(pid_map.len(), 2, "Expected 2 entries in pid_map got {pid_dump}");
247
248 let _ = manager.map_pid_to_koid(another_current.task.get_pid());
250 let _ = manager.map_pid_to_koid(another_current.task.get_tid());
251
252 manager.stop();
253
254 manager.clear();
255 assert!(manager.map.read().is_empty());
256 sender.send(()).unwrap();
257 })
258 .await;
259 receiver.await.unwrap();
260 }
261}