blob: 94df0e489250100b79614db07006668f0fdb021c [file] [log] [blame]
Hao Lu922070c2017-10-18 16:29:02 -07001#include <stdio.h>
2#include <stdint.h>
3#include <stdlib.h>
4#include <string.h>
5#include <alloca.h>
6
7#include <errno.h>
8#include <sys/types.h>
9#include <sys/sysctl.h>
Hao Lu922070c2017-10-18 16:29:02 -070010#include <mach/machine.h>
11
12#include <cpuinfo.h>
13#include <mach/api.h>
Marat Dukhand0b37602018-12-09 01:59:26 -080014#include <cpuinfo/internal-api.h>
15#include <cpuinfo/log.h>
Hao Lu922070c2017-10-18 16:29:02 -070016
17
Marat Dukhanec862142017-10-18 17:24:46 -070018struct cpuinfo_arm_isa cpuinfo_isa = {
19#if CPUINFO_ARCH_ARM
20 .thumb = true,
21 .thumb2 = true,
22 .thumbee = false,
23 .jazelle = false,
24 .armv5e = true,
25 .armv6 = true,
26 .armv6k = true,
27 .armv7 = true,
28 .vfpv2 = false,
29 .vfpv3 = true,
30 .d32 = true,
31 .wmmx = false,
32 .wmmx2 = false,
33 .neon = true,
34#endif
35#if CPUINFO_ARCH_ARM64
36 .aes = true,
37 .sha1 = true,
38 .sha2 = true,
39 .pmull = true,
40 .crc32 = true,
41#endif
42};
Marat Dukhan7b738882017-10-18 16:59:28 -070043
Marat Dukhan7d52b052018-03-18 22:57:05 -070044static uint32_t get_sys_info(int type_specifier, const char* name) {
Hao Lu922070c2017-10-18 16:29:02 -070045 size_t size = 0;
46 uint32_t result = 0;
47 int mib[2] = { CTL_HW, type_specifier };
48 if (sysctl(mib, 2, NULL, &size, NULL, 0) != 0) {
Marat Dukhan40c5f362018-12-26 20:23:13 -080049 cpuinfo_log_info("sysctl(\"%s\") failed: %s", name, strerror(errno));
Hao Lu922070c2017-10-18 16:29:02 -070050 } else if (size == sizeof(uint32_t)) {
51 sysctl(mib, 2, &result, &size, NULL, 0);
52 cpuinfo_log_debug("%s: %"PRIu32 ", size = %lu", name, result, size);
53 } else {
Marat Dukhan40c5f362018-12-26 20:23:13 -080054 cpuinfo_log_info("sysctl does not support non-integer lookup for (\"%s\")", name);
Hao Lu922070c2017-10-18 16:29:02 -070055 }
56 return result;
57}
58
Marat Dukhan40c5f362018-12-26 20:23:13 -080059static uint32_t get_sys_info_by_name(const char* type_specifier) {
Hao Lu922070c2017-10-18 16:29:02 -070060 size_t size = 0;
61 uint32_t result = 0;
62 if (sysctlbyname(type_specifier, NULL, &size, NULL, 0) != 0) {
Marat Dukhan40c5f362018-12-26 20:23:13 -080063 cpuinfo_log_info("sysctlbyname(\"%s\") failed: %s", type_specifier, strerror(errno));
Hao Lu922070c2017-10-18 16:29:02 -070064 } else if (size == sizeof(uint32_t)) {
65 sysctlbyname(type_specifier, &result, &size, NULL, 0);
66 cpuinfo_log_debug("%s: %"PRIu32 ", size = %lu", type_specifier, result, size);
67 } else {
Marat Dukhan40c5f362018-12-26 20:23:13 -080068 cpuinfo_log_info("sysctl does not support non-integer lookup for (\"%s\")", type_specifier);
Hao Lu922070c2017-10-18 16:29:02 -070069 }
70 return result;
71}
72
Marat Dukhan44dafc52018-12-26 20:19:15 -080073static enum cpuinfo_uarch decode_uarch(uint32_t cpu_family, uint32_t cpu_subtype, uint32_t core_index, uint32_t core_count) {
Hao Lu922070c2017-10-18 16:29:02 -070074 switch (cpu_family) {
75 case CPUFAMILY_ARM_SWIFT:
76 return cpuinfo_uarch_swift;
77 case CPUFAMILY_ARM_CYCLONE:
78 return cpuinfo_uarch_cyclone;
79 case CPUFAMILY_ARM_TYPHOON:
80 return cpuinfo_uarch_typhoon;
81 case CPUFAMILY_ARM_TWISTER:
82 return cpuinfo_uarch_twister;
83 case CPUFAMILY_ARM_HURRICANE:
84 return cpuinfo_uarch_hurricane;
85 case CPUFAMILY_ARM_MONSOON_MISTRAL:
Hao Lu922070c2017-10-18 16:29:02 -070086 /* 2x Monsoon + 4x Mistral cores */
87 return core_index < 2 ? cpuinfo_uarch_monsoon : cpuinfo_uarch_mistral;
Marat Dukhan44dafc52018-12-26 20:19:15 -080088#ifdef CPUFAMILY_ARM_VORTEX_TEMPEST
Marat Dukhan2d2bfa42018-12-27 10:31:36 -080089 case CPUFAMILY_ARM_VORTEX_TEMPEST:
Marat Dukhan44dafc52018-12-26 20:19:15 -080090#else
Marat Dukhan2d2bfa42018-12-27 10:31:36 -080091 case 0xe81e7ef6:
92 /* Hard-coded value for older SDKs which do not define CPUFAMILY_ARM_VORTEX_TEMPEST */
Marat Dukhan44dafc52018-12-26 20:19:15 -080093#endif
Marat Dukhan2d2bfa42018-12-27 10:31:36 -080094 /* Hexa-core: 2x Vortex + 4x Tempest; Octa-core: 4x Cortex + 4x Tempest */
95 return core_index + 4 < core_count ? cpuinfo_uarch_vortex : cpuinfo_uarch_tempest;
Hao Lu922070c2017-10-18 16:29:02 -070096 default:
97 /* Use hw.cpusubtype for detection */
98 break;
99 }
100
101 switch (cpu_subtype) {
102 case CPU_SUBTYPE_ARM_V7:
103 return cpuinfo_uarch_cortex_a8;
104 case CPU_SUBTYPE_ARM_V7F:
105 return cpuinfo_uarch_cortex_a9;
106 case CPU_SUBTYPE_ARM_V7K:
107 return cpuinfo_uarch_cortex_a7;
108 default:
109 return cpuinfo_uarch_unknown;
110 }
111}
112
113static void decode_package_name(char* package_name) {
114 size_t size;
115 if (sysctlbyname("hw.machine", NULL, &size, NULL, 0) != 0) {
116 cpuinfo_log_warning("sysctlbyname(\"hw.machine\") failed: %s", strerror(errno));
117 return;
118 }
119
120 char *machine_name = alloca(size);
121 if (sysctlbyname("hw.machine", machine_name, &size, NULL, 0) != 0) {
122 cpuinfo_log_warning("sysctlbyname(\"hw.machine\") failed: %s", strerror(errno));
123 return;
124 }
125 cpuinfo_log_debug("hw.machine: %s", machine_name);
Hao Lu8c2a3832018-07-23 23:12:11 -0700126
Hao Lu922070c2017-10-18 16:29:02 -0700127 char name[10];
128 uint32_t major = 0, minor = 0;
129 if (sscanf(machine_name, "%9[^,0123456789]%"SCNu32",%"SCNu32, name, &major, &minor) != 3) {
130 cpuinfo_log_warning("parsing \"hw.machine\" failed: %s", strerror(errno));
131 return;
132 }
Hao Lu8c2a3832018-07-23 23:12:11 -0700133
Hao Lu922070c2017-10-18 16:29:02 -0700134 uint32_t chip_model = 0;
135 char suffix = '\0';
136 if (strcmp(name, "iPhone") == 0) {
137 /*
138 * iPhone 4 and up are supported:
139 * - iPhone 4 [A4]: iPhone3,1, iPhone3,2, iPhone3,3
140 * - iPhone 4S [A5]: iPhone4,1
141 * - iPhone 5 [A6]: iPhone5,1, iPhone5,2
142 * - iPhone 5c [A6]: iPhone5,3, iPhone5,4
143 * - iPhone 5s [A7]: iPhone6,1, iPhone6,2
144 * - iPhone 6 [A8]: iPhone7,2
145 * - iPhone 6 Plus [A8]: iPhone7,1
146 * - iPhone 6s [A9]: iPhone8,1
147 * - iPhone 6s Plus [A9]: iPhone8,2
148 * - iPhone SE [A9]: iPhone8,4
149 * - iPhone 7 [A10]: iPhone9,1, iPhone9,3
150 * - iPhone 7 Plus [A10]: iPhone9,2, iPhone9,4
151 * - iPhone 8 [A11]: iPhone10,1, iPhone10,4
152 * - iPhone 8 Plus [A11]: iPhone10,2, iPhone10,5
153 * - iPhone X [A11]: iPhone10,3, iPhone10,6
Marat Dukhanfd54c3d2018-09-30 22:21:20 -0700154 * - iPhone XS [A12]: iPhone11,2,
155 * - iPhone XS Max [A12]: iPhone11,4, iPhone11,6
156 * - iPhone XR [A12]: iPhone11,8
Hao Lu922070c2017-10-18 16:29:02 -0700157 */
158 chip_model = major + 1;
159 } else if (strcmp(name, "iPad") == 0) {
160 switch (major) {
161 /* iPad 2 and up are supported */
162 case 2:
163 /*
164 * iPad 2 [A5]: iPad2,1, iPad2,2, iPad2,3, iPad2,4
165 * iPad mini [A5]: iPad2,5, iPad2,6, iPad2,7
166 */
167 chip_model = major + 3;
168 break;
169 case 3:
170 /*
171 * iPad 3rd Gen [A5X]: iPad3,1, iPad3,2, iPad3,3
172 * iPad 4th Gen [A6X]: iPad3,4, iPad3,5, iPad3,6
173 */
174 chip_model = (minor <= 3) ? 5 : 6;
175 suffix = 'X';
176 break;
177 case 4:
178 /*
179 * iPad Air [A7]: iPad4,1, iPad4,2, iPad4,3
180 * iPad mini Retina [A7]: iPad4,4, iPad4,5, iPad4,6
181 * iPad mini 3 [A7]: iPad4,7, iPad4,8, iPad4,9
182 */
183 chip_model = major + 3;
184 break;
185 case 5:
186 /*
187 * iPad mini 4 [A8]: iPad5,1, iPad5,2
188 * iPad Air 2 [A8X]: iPad5,3, iPad5,4
189 */
190 chip_model = major + 3;
191 suffix = (minor <= 2) ? '\0' : 'X';
192 break;
193 case 6:
194 /*
195 * iPad Pro 9.7" [A9X]: iPad6,3, iPad6,4
196 * iPad Pro [A9X]: iPad6,7, iPad6,8
197 * iPad 5th Gen [A9]: iPad6,11, iPad6,12
198 */
199 chip_model = major + 3;
200 suffix = minor <= 8 ? 'X' : '\0';
201 break;
202 case 7:
203 /*
204 * iPad Pro 12.9" [A10X]: iPad7,1, iPad7,2
205 * iPad Pro 10.5" [A10X]: iPad7,3, iPad7,4
Marat Dukhanfd54c3d2018-09-30 22:21:20 -0700206 * iPad 6th Gen [A10]: iPad7,5, iPad7,6
Hao Lu922070c2017-10-18 16:29:02 -0700207 */
208 chip_model = major + 3;
Marat Dukhanfd54c3d2018-09-30 22:21:20 -0700209 suffix = minor <= 4 ? 'X' : '\0';
Hao Lu922070c2017-10-18 16:29:02 -0700210 break;
211 default:
212 cpuinfo_log_info("unknown iPad: %s", machine_name);
213 break;
214 }
215 } else if (strcmp(name, "iPod") == 0) {
216 switch (major) {
217 case 5:
218 chip_model = 5;
219 break;
220 /* iPod touch (5th Gen) [A5]: iPod5,1 */
221 case 7:
222 /* iPod touch (6th Gen, 2015) [A8]: iPod7,1 */
223 chip_model = 8;
224 break;
225 default:
226 cpuinfo_log_info("unknown iPod: %s", machine_name);
227 break;
228 }
229 } else {
230 cpuinfo_log_info("unknown device: %s", machine_name);
231 }
232 if (chip_model != 0) {
233 snprintf(package_name, CPUINFO_PACKAGE_NAME_MAX, "Apple A%"PRIu32"%c", chip_model, suffix);
234 }
235}
236
237void cpuinfo_arm_mach_init(void) {
238 struct cpuinfo_processor* processors = NULL;
239 struct cpuinfo_core* cores = NULL;
Hao Lu8c2a3832018-07-23 23:12:11 -0700240 struct cpuinfo_cluster* clusters = NULL;
Hao Lu922070c2017-10-18 16:29:02 -0700241 struct cpuinfo_package* packages = NULL;
242 struct cpuinfo_cache* l1i = NULL;
243 struct cpuinfo_cache* l1d = NULL;
244 struct cpuinfo_cache* l2 = NULL;
245 struct cpuinfo_cache* l3 = NULL;
246
247 struct cpuinfo_mach_topology mach_topology = cpuinfo_mach_detect_topology();
248 processors = calloc(mach_topology.threads, sizeof(struct cpuinfo_processor));
249 if (processors == NULL) {
250 cpuinfo_log_error("failed to allocate %zu bytes for descriptions of %"PRIu32" logical processors",
251 mach_topology.threads * sizeof(struct cpuinfo_processor), mach_topology.threads);
252 goto cleanup;
253 }
254 cores = calloc(mach_topology.cores, sizeof(struct cpuinfo_core));
255 if (cores == NULL) {
256 cpuinfo_log_error("failed to allocate %zu bytes for descriptions of %"PRIu32" cores",
257 mach_topology.cores * sizeof(struct cpuinfo_core), mach_topology.cores);
258 goto cleanup;
259 }
260 packages = calloc(mach_topology.packages, sizeof(struct cpuinfo_package));
261 if (packages == NULL) {
262 cpuinfo_log_error("failed to allocate %zu bytes for descriptions of %"PRIu32" packages",
263 mach_topology.packages * sizeof(struct cpuinfo_package), mach_topology.packages);
264 goto cleanup;
265 }
266
267 const uint32_t threads_per_core = mach_topology.threads / mach_topology.cores;
268 const uint32_t threads_per_package = mach_topology.threads / mach_topology.packages;
269 const uint32_t cores_per_package = mach_topology.cores / mach_topology.packages;
Hao Lu8c2a3832018-07-23 23:12:11 -0700270
Hao Lu922070c2017-10-18 16:29:02 -0700271 for (uint32_t i = 0; i < mach_topology.packages; i++) {
272 packages[i] = (struct cpuinfo_package) {
273 .processor_start = i * threads_per_package,
274 .processor_count = threads_per_package,
275 .core_start = i * cores_per_package,
276 .core_count = cores_per_package,
277 };
278 decode_package_name(packages[i].name);
279 }
Marat Dukhanec862142017-10-18 17:24:46 -0700280
281
Hao Lu922070c2017-10-18 16:29:02 -0700282 const uint32_t cpu_family = get_sys_info_by_name("hw.cpufamily");
Marat Dukhanec862142017-10-18 17:24:46 -0700283 const uint32_t cpu_type = get_sys_info_by_name("hw.cputype");
Hao Lu922070c2017-10-18 16:29:02 -0700284 const uint32_t cpu_subtype = get_sys_info_by_name("hw.cpusubtype");
Marat Dukhanec862142017-10-18 17:24:46 -0700285 switch (cpu_type) {
286 case CPU_TYPE_ARM64:
287 cpuinfo_isa.aes = true;
288 cpuinfo_isa.sha1 = true;
289 cpuinfo_isa.sha2 = true;
290 cpuinfo_isa.pmull = true;
291 cpuinfo_isa.crc32 = true;
292 break;
293#if CPUINFO_ARCH_ARM
294 case CPU_TYPE_ARM:
295 switch (cpu_subtype) {
296 case CPU_SUBTYPE_ARM_V8:
297 cpuinfo_isa.aes = true;
298 cpuinfo_isa.sha1 = true;
299 cpuinfo_isa.sha2 = true;
300 cpuinfo_isa.pmull = true;
301 cpuinfo_isa.crc32 = true;
302 /* Fall-through to add ARMv7S features */
303 case CPU_SUBTYPE_ARM_V7S:
304 case CPU_SUBTYPE_ARM_V7K:
305 cpuinfo_isa.fma = true;
306 /* Fall-through to add ARMv7F features */
307 case CPU_SUBTYPE_ARM_V7F:
308 cpuinfo_isa.armv7mp = true;
309 cpuinfo_isa.fp16 = true;
310 /* Fall-through to add ARMv7 features */
311 case CPU_SUBTYPE_ARM_V7:
312 break;
313 default:
314 break;
315 }
316 break;
317#endif
318 }
Marat Dukhan2d2bfa42018-12-27 10:31:36 -0800319 /*
320 * Support for ARMv8.1 Atomics & FP16 arithmetic instructions is supposed to be detected via
321 * sysctlbyname calls with "hw.optional.armv8_1_atomics" and "hw.optional.neon_fp16" arguments
322 * (see https://devstreaming-cdn.apple.com/videos/wwdc/2018/409t8zw7rumablsh/409/409_whats_new_in_llvm.pdf),
323 * but on new iOS versions these calls just fail with EPERM.
324 *
325 * Thus, we whitelist CPUs known to support these instructions.
326 */
327 switch (cpu_family) {
328 case CPUFAMILY_ARM_MONSOON_MISTRAL:
Marat Dukhanc82f5e32018-12-26 20:20:23 -0800329#ifdef CPUFAMILY_ARM_VORTEX_TEMPEST
Marat Dukhan2d2bfa42018-12-27 10:31:36 -0800330 case CPUFAMILY_ARM_VORTEX_TEMPEST:
Marat Dukhanc82f5e32018-12-26 20:20:23 -0800331#else
Marat Dukhan2d2bfa42018-12-27 10:31:36 -0800332 case 0xe81e7ef6:
333 /* Hard-coded value for older SDKs which do not define CPUFAMILY_ARM_VORTEX_TEMPEST */
Marat Dukhanc82f5e32018-12-26 20:20:23 -0800334#endif
Marat Dukhan2d2bfa42018-12-27 10:31:36 -0800335#if CPUINFO_ARCH_ARM64
336 cpuinfo_isa.atomics = true;
337#endif
338 cpuinfo_isa.fp16arith = true;
339 }
Marat Dukhanec862142017-10-18 17:24:46 -0700340
Hao Lu8c2a3832018-07-23 23:12:11 -0700341 uint32_t num_clusters = 1;
Hao Lu922070c2017-10-18 16:29:02 -0700342 for (uint32_t i = 0; i < mach_topology.cores; i++) {
343 cores[i] = (struct cpuinfo_core) {
344 .processor_start = i * threads_per_core,
345 .processor_count = threads_per_core,
346 .core_id = i % cores_per_package,
347 .package = packages + i / cores_per_package,
348 .vendor = cpuinfo_vendor_apple,
Marat Dukhan44dafc52018-12-26 20:19:15 -0800349 .uarch = decode_uarch(cpu_family, cpu_subtype, i, mach_topology.cores),
Hao Lu922070c2017-10-18 16:29:02 -0700350 };
Hao Lu8c2a3832018-07-23 23:12:11 -0700351 if (i != 0 && cores[i].uarch != cores[i - 1].uarch) {
352 num_clusters++;
353 }
Hao Lu922070c2017-10-18 16:29:02 -0700354 }
355 for (uint32_t i = 0; i < mach_topology.threads; i++) {
356 const uint32_t smt_id = i % threads_per_core;
357 const uint32_t core_id = i / threads_per_core;
358 const uint32_t package_id = i / threads_per_package;
359
360 processors[i].smt_id = smt_id;
Marat Dukhan7fcd4412017-11-30 09:46:49 -0800361 processors[i].core = &cores[core_id];
362 processors[i].package = &packages[package_id];
Hao Lu922070c2017-10-18 16:29:02 -0700363 }
364
Hao Lu8c2a3832018-07-23 23:12:11 -0700365 clusters = calloc(num_clusters, sizeof(struct cpuinfo_cluster));
366 if (clusters == NULL) {
367 cpuinfo_log_error(
368 "failed to allocate %zu bytes for descriptions of %"PRIu32" clusters",
369 num_clusters * sizeof(struct cpuinfo_cluster), num_clusters);
370 goto cleanup;
371 }
372 uint32_t cluster_idx = UINT32_MAX;
373 for (uint32_t i = 0; i < mach_topology.cores; i++) {
374 if (i == 0 || cores[i].uarch != cores[i - 1].uarch) {
375 cluster_idx++;
376 clusters[cluster_idx] = (struct cpuinfo_cluster) {
377 .processor_start = i * threads_per_core,
378 .processor_count = 1,
379 .core_start = i,
380 .core_count = 1,
381 .cluster_id = cluster_idx,
382 .package = cores[i].package,
383 .vendor = cores[i].vendor,
384 .uarch = cores[i].uarch,
385 };
386 } else {
387 clusters[cluster_idx].processor_count++;
388 clusters[cluster_idx].core_count++;
389 }
390 cores[i].cluster = &clusters[cluster_idx];
391 }
392
393 for (uint32_t i = 0; i < mach_topology.threads; i++) {
394 const uint32_t core_id = i / threads_per_core;
395 processors[i].cluster = cores[core_id].cluster;
396 }
397
398 for (uint32_t i = 0; i < mach_topology.packages; i++) {
399 packages[i].cluster_start = 0;
400 packages[i].cluster_count = num_clusters;
401 }
402
Hao Lu922070c2017-10-18 16:29:02 -0700403 const uint32_t cacheline_size = get_sys_info(HW_CACHELINE, "HW_CACHELINE");
404 const uint32_t l1d_cache_size = get_sys_info(HW_L1DCACHESIZE, "HW_L1DCACHESIZE");
405 const uint32_t l1i_cache_size = get_sys_info(HW_L1ICACHESIZE, "HW_L1ICACHESIZE");
406 const uint32_t l2_cache_size = get_sys_info(HW_L2CACHESIZE, "HW_L2CACHESIZE");
407 const uint32_t l3_cache_size = get_sys_info(HW_L3CACHESIZE, "HW_L3CACHESIZE");
408 const uint32_t l1_cache_associativity = 4;
409 const uint32_t l2_cache_associativity = 8;
410 const uint32_t l3_cache_associativity = 16;
411 const uint32_t cache_partitions = 1;
412 const uint32_t cache_flags = 0;
413
414 uint32_t threads_per_l1 = 0, l1_count = 0;
415 if (l1i_cache_size != 0 || l1d_cache_size != 0) {
Hao Lu3617d5b2017-10-23 15:16:50 -0700416 /* Assume L1 caches are private to each core */
Hao Lu922070c2017-10-18 16:29:02 -0700417 threads_per_l1 = 1;
418 l1_count = mach_topology.threads / threads_per_l1;
419 cpuinfo_log_debug("detected %"PRIu32" L1 caches", l1_count);
420 }
421
422 uint32_t threads_per_l2 = 0, l2_count = 0;
423 if (l2_cache_size != 0) {
Hao Lu3617d5b2017-10-23 15:16:50 -0700424 /* Assume L2 cache is shared between all cores */
Hao Lu922070c2017-10-18 16:29:02 -0700425 threads_per_l2 = mach_topology.cores;
426 l2_count = 1;
427 cpuinfo_log_debug("detected %"PRIu32" L2 caches", l2_count);
428 }
Hao Lu8c2a3832018-07-23 23:12:11 -0700429
Hao Lu922070c2017-10-18 16:29:02 -0700430 uint32_t threads_per_l3 = 0, l3_count = 0;
431 if (l3_cache_size != 0) {
Hao Lu3617d5b2017-10-23 15:16:50 -0700432 /* Assume L3 cache is shared between all cores */
Hao Lu922070c2017-10-18 16:29:02 -0700433 threads_per_l3 = mach_topology.cores;
434 l3_count = 1;
435 cpuinfo_log_debug("detected %"PRIu32" L3 caches", l3_count);
436 }
437
438 if (l1i_cache_size != 0) {
439 l1i = calloc(l1_count, sizeof(struct cpuinfo_cache));
440 if (l1i == NULL) {
441 cpuinfo_log_error("failed to allocate %zu bytes for descriptions of %"PRIu32" L1I caches",
442 l1_count * sizeof(struct cpuinfo_cache), l1_count);
443 goto cleanup;
444 }
445 for (uint32_t c = 0; c < l1_count; c++) {
446 l1i[c] = (struct cpuinfo_cache) {
447 .size = l1i_cache_size,
448 .associativity = l1_cache_associativity,
449 .sets = l1i_cache_size / (l1_cache_associativity * cacheline_size),
450 .partitions = cache_partitions,
451 .line_size = cacheline_size,
452 .flags = cache_flags,
453 .processor_start = c * threads_per_l1,
454 .processor_count = threads_per_l1,
455 };
456 }
457 for (uint32_t t = 0; t < mach_topology.threads; t++) {
458 processors[t].cache.l1i = &l1i[t / threads_per_l1];
459 }
460 }
461
462 if (l1d_cache_size != 0) {
463 l1d = calloc(l1_count, sizeof(struct cpuinfo_cache));
464 if (l1d == NULL) {
465 cpuinfo_log_error("failed to allocate %zu bytes for descriptions of %"PRIu32" L1D caches",
466 l1_count * sizeof(struct cpuinfo_cache), l1_count);
467 goto cleanup;
468 }
469 for (uint32_t c = 0; c < l1_count; c++) {
470 l1d[c] = (struct cpuinfo_cache) {
471 .size = l1d_cache_size,
472 .associativity = l1_cache_associativity,
473 .sets = l1d_cache_size / (l1_cache_associativity * cacheline_size),
474 .partitions = cache_partitions,
475 .line_size = cacheline_size,
476 .flags = cache_flags,
477 .processor_start = c * threads_per_l1,
478 .processor_count = threads_per_l1,
479 };
480 }
481 for (uint32_t t = 0; t < mach_topology.threads; t++) {
482 processors[t].cache.l1d = &l1d[t / threads_per_l1];
483 }
484 }
485
486 if (l2_count != 0) {
487 l2 = calloc(l2_count, sizeof(struct cpuinfo_cache));
488 if (l2 == NULL) {
489 cpuinfo_log_error("failed to allocate %zu bytes for descriptions of %"PRIu32" L2 caches",
490 l2_count * sizeof(struct cpuinfo_cache), l2_count);
491 goto cleanup;
492 }
493 for (uint32_t c = 0; c < l2_count; c++) {
494 l2[c] = (struct cpuinfo_cache) {
495 .size = l2_cache_size,
496 .associativity = l2_cache_associativity,
497 .sets = l2_cache_size / (l2_cache_associativity * cacheline_size),
498 .partitions = cache_partitions,
499 .line_size = cacheline_size,
500 .flags = cache_flags,
501 .processor_start = c * threads_per_l2,
502 .processor_count = threads_per_l2,
503 };
504 }
505 for (uint32_t t = 0; t < mach_topology.threads; t++) {
506 processors[t].cache.l2 = &l2[0];
507 }
508 }
Hao Lu8c2a3832018-07-23 23:12:11 -0700509
Hao Lu922070c2017-10-18 16:29:02 -0700510 if (l3_count != 0) {
511 l3 = calloc(l3_count, sizeof(struct cpuinfo_cache));
512 if (l3 == NULL) {
513 cpuinfo_log_error("failed to allocate %zu bytes for descriptions of %"PRIu32" L3 caches",
514 l3_count * sizeof(struct cpuinfo_cache), l3_count);
515 goto cleanup;
516 }
517 for (uint32_t c = 0; c < l3_count; c++) {
518 l3[c] = (struct cpuinfo_cache) {
519 .size = l3_cache_size,
520 .associativity = l3_cache_associativity,
521 .sets = l3_cache_size / (l3_cache_associativity * cacheline_size),
522 .partitions = cache_partitions,
523 .line_size = cacheline_size,
524 .flags = cache_flags,
525 .processor_start = c * threads_per_l3,
526 .processor_count = threads_per_l3,
527 };
528 }
529 for (uint32_t t = 0; t < mach_topology.threads; t++) {
530 processors[t].cache.l3 = &l3[0];
531 }
532 }
533
534 /* Commit changes */
535 cpuinfo_cache[cpuinfo_cache_level_1i] = l1i;
536 cpuinfo_cache[cpuinfo_cache_level_1d] = l1d;
537 cpuinfo_cache[cpuinfo_cache_level_2] = l2;
538 cpuinfo_cache[cpuinfo_cache_level_3] = l3;
539
540 cpuinfo_processors = processors;
541 cpuinfo_cores = cores;
Hao Lu8c2a3832018-07-23 23:12:11 -0700542 cpuinfo_clusters = clusters;
Hao Lu922070c2017-10-18 16:29:02 -0700543 cpuinfo_packages = packages;
544
545 cpuinfo_cache_count[cpuinfo_cache_level_1i] = l1_count;
546 cpuinfo_cache_count[cpuinfo_cache_level_1d] = l1_count;
547 cpuinfo_cache_count[cpuinfo_cache_level_2] = l2_count;
548 cpuinfo_cache_count[cpuinfo_cache_level_3] = l3_count;
549
550 cpuinfo_processors_count = mach_topology.threads;
551 cpuinfo_cores_count = mach_topology.cores;
Hao Lu8c2a3832018-07-23 23:12:11 -0700552 cpuinfo_clusters_count = num_clusters;
Hao Lu922070c2017-10-18 16:29:02 -0700553 cpuinfo_packages_count = mach_topology.packages;
554
Marat Dukhancf70aee2018-03-24 23:21:02 -0700555 __sync_synchronize();
556
557 cpuinfo_is_initialized = true;
558
Hao Lu922070c2017-10-18 16:29:02 -0700559 processors = NULL;
560 cores = NULL;
Hao Lu8c2a3832018-07-23 23:12:11 -0700561 clusters = NULL;
Hao Lu922070c2017-10-18 16:29:02 -0700562 packages = NULL;
563 l1i = l1d = l2 = l3 = NULL;
564
565cleanup:
566 free(processors);
567 free(cores);
Hao Lu8c2a3832018-07-23 23:12:11 -0700568 free(clusters);
Hao Lu922070c2017-10-18 16:29:02 -0700569 free(packages);
570 free(l1i);
571 free(l1d);
572 free(l2);
573 free(l3);
574}