use memflow::prelude::v1::*; use super::UtlMemoryPoolBase; use crate::error::Result; use crate::mem::PointerExt; #[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 HashFixedDataInternal { pub ui_key: K, // 0x0000 pub next: Pointer64>, // 0x0008 pub data: D, // 0x0010 } unsafe impl Pod for HashFixedDataInternal {} #[repr(C)] pub struct UtlTsHash { pub entry_mem: UtlMemoryPoolBase, // 0x0000 pub buckets: [HashBucket; C], // 0x0080 pub needs_commit: bool, // 0x2880 pad_2881: [u8; 0xF], // 0x2881 } impl UtlTsHash where D: Pod + PointerExt, K: Pod, { /// Returns the number of allocated blocks. #[inline] pub fn blocks_alloc(&self) -> i32 { self.entry_mem.blocks_alloc } /// Returns the size of a block. #[inline] pub fn block_size(&self) -> i32 { self.entry_mem.block_size } /// Returns the maximum number of allocated blocks. #[inline] pub fn peak_count(&self) -> i32 { self.entry_mem.peak_alloc } /// Returns a list of allocated or unallocated elements. pub fn elements(&self, process: &mut IntoProcessInstanceArcBox<'_>) -> Result> { let blocks_alloc = self.blocks_alloc() as usize; let peak_alloc = self.peak_count() as usize; let mut allocated_list = Vec::with_capacity(peak_alloc); let mut unallocated_list = Vec::with_capacity(blocks_alloc); for bucket in &self.buckets { let mut cur_element = bucket.first_uncommitted; while !cur_element.is_null() { let element = cur_element.read(process)?; if !element.data.is_null() { unallocated_list.push(element.data); } if unallocated_list.len() >= blocks_alloc { break; } cur_element = element.next; } } let mut cur_blob = Pointer64::>::from(self.entry_mem.free_list_head.address()); while !cur_blob.is_null() { let blob = cur_blob.read(process)?; if !blob.data.is_null() { allocated_list.push(blob.data); } if allocated_list.len() >= peak_alloc { break; } cur_blob = blob.next; } Ok(if unallocated_list.len() > allocated_list.len() { unallocated_list } else { allocated_list }) } } unsafe impl Pod for UtlTsHash {}