use memflow::prelude::v1::*; use super::UtlMemoryPool; use crate::error::Result; use crate::mem::IsNull; #[repr(C)] pub struct HashAllocatedBlob { pub next: Pointer64>, // 0x0000 pad_0008: [u8; 0x8], // 0x0008 pub data: D, // 0x0010 pad_0018: [u8; 0x8], // 0x0018 } unsafe impl Pod for HashAllocatedBlob {} #[repr(C)] pub struct HashBucket { pad_0000: [u8; 0x18], // 0x0000, pub first: Pointer64>, // 0x0018 pub first_uncommitted: Pointer64>, // 0x0020 } #[repr(C)] pub struct HashFixedData { pub ui_key: K, // 0x0000 pub next: Pointer64>, // 0x0008 pub data: D, // 0x0010 } unsafe impl Pod for HashFixedData {} /// Represents a thread-safe hash table. #[repr(C)] pub struct UtlTsHash { pub entry_mem: UtlMemoryPool, // 0x0000 pad_0018: [u8; 0x8], // 0x0018 pub blobs: Pointer64>, // 0x0020 pad_0028: [u8; 0x58], // 0x0028 pub buckets: [HashBucket; C], // 0x0080 pad_2880: [u8; 0x10], // 0x2880 } impl UtlTsHash { /// Returns all elements in the hash table. pub fn elements(&self, process: &mut IntoProcessInstanceArcBox<'_>) -> Result> { // TODO: Refactor this. let mut elements: Vec<_> = self .buckets .iter() .flat_map(|bucket| { let mut cur_element = bucket.first; let mut list = Vec::new(); while !cur_element.is_null() { if let Ok(element) = cur_element.read(process) { if !element.data.is_null() { list.push(element.data); } cur_element = element.next; } } list }) .collect(); if let Ok(blob) = self.blobs.read(process) { let mut unallocated_data = blob.next; if !unallocated_data.is_null() { if !blob.data.is_null() { elements.push(blob.data); } while !unallocated_data.is_null() { if let Ok(element) = unallocated_data.read(process) { if !element.data.is_null() { elements.push(element.data); } unallocated_data = element.next; } } } } Ok(elements) } } unsafe impl Pod for UtlTsHash {}