1*a325d9c4SApple OSS Distributions #include <darwintest.h> 2*a325d9c4SApple OSS Distributions #include <inttypes.h> 3*a325d9c4SApple OSS Distributions #if __arm64__ 4*a325d9c4SApple OSS Distributions #include <mach/arm/processor_info.h> 5*a325d9c4SApple OSS Distributions #endif /* __arm64__ */ 6*a325d9c4SApple OSS Distributions #include <mach/mach.h> 7*a325d9c4SApple OSS Distributions #include <stdlib.h> 8*a325d9c4SApple OSS Distributions #include <unistd.h> 9*a325d9c4SApple OSS Distributions 10*a325d9c4SApple OSS Distributions T_GLOBAL_META(T_META_ASROOT(true), 11*a325d9c4SApple OSS Distributions T_META_RUN_CONCURRENTLY(true)); 12*a325d9c4SApple OSS Distributions 13*a325d9c4SApple OSS Distributions T_DECL(processor_cpu_stat64, 14*a325d9c4SApple OSS Distributions "ensure 64-bit processor statistics are reported correctly", 15*a325d9c4SApple OSS Distributions T_META_NAMESPACE("xnu.arm"), 16*a325d9c4SApple OSS Distributions T_META_RADAR_COMPONENT_NAME("xnu"), 17*a325d9c4SApple OSS Distributions T_META_RADAR_COMPONENT_VERSION("arm"), 18*a325d9c4SApple OSS Distributions T_META_OWNER("mwm")) 19*a325d9c4SApple OSS Distributions { 20*a325d9c4SApple OSS Distributions #if !__arm64__ 21*a325d9c4SApple OSS Distributions T_SKIP("processor statistics only available on ARM"); 22*a325d9c4SApple OSS Distributions #else /* !__arm64__ */ 23*a325d9c4SApple OSS Distributions host_t host = mach_host_self(); 24*a325d9c4SApple OSS Distributions host_t priv_port = MACH_PORT_NULL; 25*a325d9c4SApple OSS Distributions 26*a325d9c4SApple OSS Distributions kern_return_t kr = host_get_host_priv_port(host, &priv_port); 27*a325d9c4SApple OSS Distributions T_QUIET; 28*a325d9c4SApple OSS Distributions T_ASSERT_MACH_SUCCESS(kr, "host_get_host_priv_port"); 29*a325d9c4SApple OSS Distributions T_QUIET; 30*a325d9c4SApple OSS Distributions T_ASSERT_NE(priv_port, MACH_PORT_NULL, "valid host priv port"); 31*a325d9c4SApple OSS Distributions 32*a325d9c4SApple OSS Distributions processor_port_array_t cpu_ports = NULL; 33*a325d9c4SApple OSS Distributions mach_msg_type_number_t cpu_count = 0; 34*a325d9c4SApple OSS Distributions kr = host_processors(priv_port, &cpu_ports, &cpu_count); 35*a325d9c4SApple OSS Distributions T_QUIET; 36*a325d9c4SApple OSS Distributions T_ASSERT_MACH_SUCCESS(kr, "host_processors"); 37*a325d9c4SApple OSS Distributions T_QUIET; 38*a325d9c4SApple OSS Distributions T_ASSERT_NOTNULL(cpu_ports, "valid processor port array"); 39*a325d9c4SApple OSS Distributions T_QUIET; 40*a325d9c4SApple OSS Distributions T_ASSERT_GT(cpu_count, (mach_msg_type_number_t)0, 41*a325d9c4SApple OSS Distributions "non-zero CPU count"); 42*a325d9c4SApple OSS Distributions 43*a325d9c4SApple OSS Distributions T_LOG("found %d CPUs", cpu_count); 44*a325d9c4SApple OSS Distributions 45*a325d9c4SApple OSS Distributions struct processor_cpu_stat64 *prestats = calloc(cpu_count, 46*a325d9c4SApple OSS Distributions sizeof(*prestats)); 47*a325d9c4SApple OSS Distributions T_WITH_ERRNO; 48*a325d9c4SApple OSS Distributions T_QUIET; 49*a325d9c4SApple OSS Distributions T_ASSERT_NOTNULL(prestats, "allocate space for stats (pre)"); 50*a325d9c4SApple OSS Distributions memset(prestats, 0xff, cpu_count * sizeof(*prestats)); 51*a325d9c4SApple OSS Distributions 52*a325d9c4SApple OSS Distributions for (int i = 0; i < (int)cpu_count; i++) { 53*a325d9c4SApple OSS Distributions mach_msg_type_number_t info_count = PROCESSOR_CPU_STAT64_COUNT; 54*a325d9c4SApple OSS Distributions kr = processor_info(cpu_ports[i], PROCESSOR_CPU_STAT64, &host, 55*a325d9c4SApple OSS Distributions (processor_info_t)&prestats[i], &info_count); 56*a325d9c4SApple OSS Distributions T_ASSERT_MACH_SUCCESS(kr, 57*a325d9c4SApple OSS Distributions "processor_info(%d, PROCESSOR_CPU_STAT64, ...)", i); 58*a325d9c4SApple OSS Distributions 59*a325d9c4SApple OSS Distributions T_QUIET; 60*a325d9c4SApple OSS Distributions T_ASSERT_EQ(info_count, PROCESSOR_CPU_STAT64_COUNT, 61*a325d9c4SApple OSS Distributions "received enough CPU statistics"); 62*a325d9c4SApple OSS Distributions } 63*a325d9c4SApple OSS Distributions 64*a325d9c4SApple OSS Distributions sleep(1); 65*a325d9c4SApple OSS Distributions 66*a325d9c4SApple OSS Distributions struct processor_cpu_stat64 *poststats = calloc(cpu_count - 1, 67*a325d9c4SApple OSS Distributions sizeof(*poststats)); 68*a325d9c4SApple OSS Distributions T_WITH_ERRNO; 69*a325d9c4SApple OSS Distributions T_QUIET; 70*a325d9c4SApple OSS Distributions T_ASSERT_NOTNULL(poststats, "allocate space for stats (post)"); 71*a325d9c4SApple OSS Distributions 72*a325d9c4SApple OSS Distributions for (int i = 0; i < (int)cpu_count; i++) { 73*a325d9c4SApple OSS Distributions mach_msg_type_number_t info_count = PROCESSOR_CPU_STAT64_COUNT; 74*a325d9c4SApple OSS Distributions kr = processor_info(cpu_ports[i], PROCESSOR_CPU_STAT64, &host, 75*a325d9c4SApple OSS Distributions (processor_info_t)&poststats[i], &info_count); 76*a325d9c4SApple OSS Distributions T_ASSERT_MACH_SUCCESS(kr, 77*a325d9c4SApple OSS Distributions "processor_info(%d, PROCESSOR_CPU_STAT64, ...)", i); 78*a325d9c4SApple OSS Distributions 79*a325d9c4SApple OSS Distributions T_QUIET; 80*a325d9c4SApple OSS Distributions T_ASSERT_EQ(info_count, PROCESSOR_CPU_STAT64_COUNT, 81*a325d9c4SApple OSS Distributions "received enough CPU statistics"); 82*a325d9c4SApple OSS Distributions } 83*a325d9c4SApple OSS Distributions 84*a325d9c4SApple OSS Distributions for (int i = 0; i < (int)cpu_count; i++) { 85*a325d9c4SApple OSS Distributions #define CHECK_STAT_FIELD(field) \ 86*a325d9c4SApple OSS Distributions T_EXPECT_GE(poststats[i].field, prestats[i].field, \ 87*a325d9c4SApple OSS Distributions "CPU %d's " #field " is monotonically increasing (+%" PRIu64 \ 88*a325d9c4SApple OSS Distributions ")", i, poststats[i].field - prestats[i].field) 89*a325d9c4SApple OSS Distributions 90*a325d9c4SApple OSS Distributions CHECK_STAT_FIELD(irq_ex_cnt); 91*a325d9c4SApple OSS Distributions CHECK_STAT_FIELD(ipi_cnt); 92*a325d9c4SApple OSS Distributions CHECK_STAT_FIELD(timer_cnt); 93*a325d9c4SApple OSS Distributions CHECK_STAT_FIELD(undef_ex_cnt); 94*a325d9c4SApple OSS Distributions CHECK_STAT_FIELD(unaligned_cnt); 95*a325d9c4SApple OSS Distributions CHECK_STAT_FIELD(vfp_cnt); 96*a325d9c4SApple OSS Distributions CHECK_STAT_FIELD(vfp_shortv_cnt); 97*a325d9c4SApple OSS Distributions CHECK_STAT_FIELD(data_ex_cnt); 98*a325d9c4SApple OSS Distributions CHECK_STAT_FIELD(instr_ex_cnt); 99*a325d9c4SApple OSS Distributions CHECK_STAT_FIELD(pmi_cnt); 100*a325d9c4SApple OSS Distributions 101*a325d9c4SApple OSS Distributions #undef CHECK_STAT_FIELD 102*a325d9c4SApple OSS Distributions } 103*a325d9c4SApple OSS Distributions 104*a325d9c4SApple OSS Distributions free(prestats); 105*a325d9c4SApple OSS Distributions free(poststats); 106*a325d9c4SApple OSS Distributions #endif /* __arm64__ */ 107*a325d9c4SApple OSS Distributions } 108*a325d9c4SApple OSS Distributions 109*a325d9c4SApple OSS Distributions 110*a325d9c4SApple OSS Distributions T_DECL(processor_cpu_info_order, 111*a325d9c4SApple OSS Distributions "ensure host_processor_info iterates CPU in CPU ID order") 112*a325d9c4SApple OSS Distributions { 113*a325d9c4SApple OSS Distributions host_t host = mach_host_self(); 114*a325d9c4SApple OSS Distributions host_t priv_port = MACH_PORT_NULL; 115*a325d9c4SApple OSS Distributions 116*a325d9c4SApple OSS Distributions kern_return_t kr = host_get_host_priv_port(host, &priv_port); 117*a325d9c4SApple OSS Distributions T_QUIET; T_ASSERT_MACH_SUCCESS(kr, "host_get_host_priv_port"); 118*a325d9c4SApple OSS Distributions T_QUIET; T_ASSERT_NE(priv_port, MACH_PORT_NULL, "valid host priv port"); 119*a325d9c4SApple OSS Distributions 120*a325d9c4SApple OSS Distributions processor_info_array_t info_array = NULL; 121*a325d9c4SApple OSS Distributions mach_msg_type_number_t info_count = 0; 122*a325d9c4SApple OSS Distributions natural_t processor_count = 0; 123*a325d9c4SApple OSS Distributions 124*a325d9c4SApple OSS Distributions kr = host_processor_info(mach_host_self(), PROCESSOR_BASIC_INFO, &processor_count, 125*a325d9c4SApple OSS Distributions &info_array, &info_count); 126*a325d9c4SApple OSS Distributions 127*a325d9c4SApple OSS Distributions T_QUIET; T_ASSERT_MACH_SUCCESS(kr, "host_processor_info(PROCESSOR_BASIC_INFO)"); 128*a325d9c4SApple OSS Distributions T_QUIET; T_ASSERT_NOTNULL(info_array, "valid processor port array"); 129*a325d9c4SApple OSS Distributions T_QUIET; T_ASSERT_GT(info_count, (mach_msg_type_number_t)0, "non-zero array"); 130*a325d9c4SApple OSS Distributions T_QUIET; T_ASSERT_GT(processor_count, (natural_t)0, "non-zero processor_count"); 131*a325d9c4SApple OSS Distributions 132*a325d9c4SApple OSS Distributions processor_basic_info_t basic_info_array = (processor_basic_info_t)info_array; 133*a325d9c4SApple OSS Distributions 134*a325d9c4SApple OSS Distributions for (natural_t i = 0; i < processor_count; i++) { 135*a325d9c4SApple OSS Distributions struct processor_basic_info* processor_info = &basic_info_array[i]; 136*a325d9c4SApple OSS Distributions 137*a325d9c4SApple OSS Distributions natural_t slot_num = (natural_t)processor_info->slot_num; 138*a325d9c4SApple OSS Distributions 139*a325d9c4SApple OSS Distributions T_ASSERT_EQ(slot_num, i, "CPU ID must equal array index"); 140*a325d9c4SApple OSS Distributions } 141*a325d9c4SApple OSS Distributions } 142