// Copyright 2023 Brian Cook (a.k.a. Coding-Badly) // // Licensed under the Apache License, Version 2.0 (the "License"); // you may not use this file except in compliance with the License. // You may obtain a copy of the License at // // http://www.apache.org/licenses/LICENSE-2.0 // // Unless required by applicable law or agreed to in writing, software // distributed under the License is distributed on an "AS IS" BASIS, // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. // See the License for the specific language governing permissions and // limitations under the License. use windows::Win32::System::SystemInformation::{ GetLogicalProcessorInformationEx, RelationGroup, SYSTEM_LOGICAL_PROCESSOR_INFORMATION_EX, }; use grob::{GrowForSmallBinary, GrowableBuffer, RvIsError, StackBuffer, ToResult, WriteBuffer}; fn common(initial_buffer: &mut dyn WriteBuffer) -> Result<(), Box> { let grow_strategy = GrowForSmallBinary::new(); // Loop until the call to GetAdaptersAddresses fails with an error or succeeds because the // buffer has enough space. let mut growable_buffer = GrowableBuffer::< SYSTEM_LOGICAL_PROCESSOR_INFORMATION_EX, *mut SYSTEM_LOGICAL_PROCESSOR_INFORMATION_EX, >::new(initial_buffer, &grow_strategy); loop { // Prepare the argument for the API calll let mut argument = growable_buffer.argument(); // Make the API call indicating what the return value means let rv = RvIsError::new(unsafe { GetLogicalProcessorInformationEx( RelationGroup, Some(argument.pointer()), argument.size(), ) }); // Convert the return value to an action let fill_buffer_action = rv.to_result(&mut argument)?; // Apply the action if argument.apply(fill_buffer_action) { break; } } // Do something with the data let frozen_buffer = growable_buffer.freeze(); if let Some(p) = frozen_buffer.pointer() { let r = unsafe { (*p).Relationship }; println!("Relationship = {:?}", r); // Has to be RelationGroup println!("Size = {:?}", unsafe { (*p).Size }); println!("MaximumGroupCount = {:?}", unsafe { (*p).Anonymous.Group.MaximumGroupCount }); println!("ActiveGroupCount = {:?}", unsafe { (*p).Anonymous.Group.ActiveGroupCount }); println!("ActiveProcessorCount = {:?}", unsafe { (*p).Anonymous.Group.GroupInfo[0].ActiveProcessorCount }); println!("MaximumProcessorCount = {:?}", unsafe { (*p).Anonymous.Group.GroupInfo[0].MaximumProcessorCount }); println!(); } Ok(()) } fn just_heap_buffer() -> Result<(), Box> { let mut initial_buffer = StackBuffer::<0>::new(); common(&mut initial_buffer) } fn start_with_stack_buffer() -> Result<(), Box> { let mut initial_buffer = StackBuffer::<1024>::new(); common(&mut initial_buffer) } fn main() -> Result<(), Box> { println!(); start_with_stack_buffer()?; println!(); just_heap_buffer()?; println!(); println!(); Ok(()) }