/* * Copyright (c) 2024, Andrew Kaster * * SPDX-License-Identifier: BSD-2-Clause */ #include #if !defined(AK_OS_MACH) # error "This file is only available on Mach platforms" #endif #include #include #include #include #include namespace Core::Platform { static auto user_hz = sysconf(_SC_CLK_TCK); ErrorOr update_process_statistics(ProcessStatistics& statistics) { host_cpu_load_info_data_t cpu_info {}; mach_msg_type_number_t count = HOST_CPU_LOAD_INFO_COUNT; auto res = host_statistics(mach_host_self(), HOST_CPU_LOAD_INFO, reinterpret_cast(&cpu_info), &count); if (res != KERN_SUCCESS) { dbgln("Failed to get host statistics: {}", mach_error_string(res)); return Core::mach_error_to_error(res); } u64 total_cpu_ticks = 0; total_cpu_ticks += cpu_info.cpu_ticks[CPU_STATE_USER]; total_cpu_ticks += cpu_info.cpu_ticks[CPU_STATE_SYSTEM]; total_cpu_ticks += cpu_info.cpu_ticks[CPU_STATE_NICE]; total_cpu_ticks += cpu_info.cpu_ticks[CPU_STATE_IDLE]; auto const total_cpu_ticks_diff = total_cpu_ticks - statistics.total_time_scheduled; auto const total_cpu_seconds_diff = total_cpu_ticks_diff / (static_cast(user_hz)); auto const total_cpu_micro_diff = total_cpu_seconds_diff * 1'000'000; statistics.total_time_scheduled = total_cpu_ticks; for (auto& process : statistics.processes) { mach_task_basic_info_data_t basic_info {}; count = MACH_TASK_BASIC_INFO_COUNT; res = task_info(process->child_task_port.port(), MACH_TASK_BASIC_INFO, reinterpret_cast(&basic_info), &count); if (res != KERN_SUCCESS) { dbgln("Failed to get task info for pid {}: {}", process->pid, mach_error_string(res)); return Core::mach_error_to_error(res); } process->memory_usage_bytes = basic_info.resident_size; task_thread_times_info_data_t time_info {}; count = TASK_THREAD_TIMES_INFO_COUNT; res = task_info(process->child_task_port.port(), TASK_THREAD_TIMES_INFO, reinterpret_cast(&time_info), &count); if (res != KERN_SUCCESS) { dbgln("Failed to get thread times info for pid {}: {}", process->pid, mach_error_string(res)); return Core::mach_error_to_error(res); } timeval scratch_timeval = { static_cast(time_info.user_time.seconds), static_cast(time_info.user_time.microseconds) }; auto time_in_process = AK::Duration::from_timeval(scratch_timeval); scratch_timeval = { static_cast(time_info.system_time.seconds), static_cast(time_info.system_time.microseconds) }; time_in_process += AK::Duration::from_timeval(scratch_timeval); auto time_diff_process = time_in_process - AK::Duration::from_microseconds(process->time_spent_in_process); process->time_spent_in_process = time_in_process.to_microseconds(); process->cpu_percent = 0.0f; if (time_diff_process > AK::Duration::zero()) process->cpu_percent = 100.0f * static_cast(time_diff_process.to_microseconds()) / total_cpu_micro_diff; } return {}; } MachPort register_with_mach_server(ByteString const& server_name) { auto server_port_or_error = Core::MachPort::look_up_from_bootstrap_server(server_name); if (server_port_or_error.is_error()) { dbgln("Failed to lookup server port: {}", server_port_or_error.error()); VERIFY_NOT_REACHED(); } auto server_port = server_port_or_error.release_value(); // Send our own task port to the server so they can query statistics about us MessageWithSelfTaskPort message {}; message.header.msgh_bits = MACH_MSGH_BITS(MACH_MSG_TYPE_COPY_SEND, MACH_MSGH_BITS_ZERO) | MACH_MSGH_BITS_COMPLEX; message.header.msgh_size = sizeof(message); message.header.msgh_remote_port = server_port.port(); message.header.msgh_local_port = MACH_PORT_NULL; message.header.msgh_id = SELF_TASK_PORT_MESSAGE_ID; message.body.msgh_descriptor_count = 1; message.port_descriptor.name = mach_task_self(); message.port_descriptor.disposition = MACH_MSG_TYPE_COPY_SEND; message.port_descriptor.type = MACH_MSG_PORT_DESCRIPTOR; mach_msg_timeout_t const timeout = 100; // milliseconds auto const send_result = mach_msg(&message.header, MACH_SEND_MSG | MACH_SEND_TIMEOUT, message.header.msgh_size, 0, MACH_PORT_NULL, timeout, MACH_PORT_NULL); if (send_result != KERN_SUCCESS) { dbgln("Failed to send message to server: {}", mach_error_string(send_result)); VERIFY_NOT_REACHED(); } return server_port; } }