blob: b54f5333465dced689afcdc6a4844ab1bbb881a8 [file] [log] [blame]
Marat Dukhan3045d4f2017-03-04 01:51:42 -05001#include <stdint.h>
2#include <stddef.h>
3#include <stdlib.h>
4#include <string.h>
Marat Dukhan3045d4f2017-03-04 01:51:42 -05005
6#include <cpuinfo.h>
7#include <x86/api.h>
Marat Dukhan4f70b9a2017-09-25 23:18:41 -07008#include <x86/linux/api.h>
Marat Dukhanfbc4cfc2017-09-22 16:45:45 -07009#include <gpu/api.h>
Marat Dukhan3045d4f2017-03-04 01:51:42 -050010#include <linux/api.h>
11#include <api.h>
12#include <log.h>
13
14
Marat Dukhan77eb9412017-03-04 21:01:16 -050015static inline uint32_t bit_mask(uint32_t bits) {
16 return (UINT32_C(1) << bits) - UINT32_C(1);
17}
18
Marat Dukhan4f70b9a2017-09-25 23:18:41 -070019static inline bool bitmask_all(uint32_t bitfield, uint32_t mask) {
20 return (bitfield & mask) == mask;
21}
Marat Dukhan3045d4f2017-03-04 01:51:42 -050022
Marat Dukhan4f70b9a2017-09-25 23:18:41 -070023static inline uint32_t min(uint32_t a, uint32_t b) {
24 return a < b ? a : b;
25}
26
27static inline int cmp(uint32_t a, uint32_t b) {
28 return (a > b) - (a < b);
29}
30
31static int cmp_x86_linux_processor(const void* ptr_a, const void* ptr_b) {
32 const struct cpuinfo_x86_linux_processor* processor_a = (const struct cpuinfo_x86_linux_processor*) ptr_a;
33 const struct cpuinfo_x86_linux_processor* processor_b = (const struct cpuinfo_x86_linux_processor*) ptr_b;
34
35 /* Move usable processors towards the start of the array */
36 const bool usable_a = bitmask_all(processor_a->flags, CPUINFO_LINUX_MASK_USABLE);
37 const bool usable_b = bitmask_all(processor_b->flags, CPUINFO_LINUX_MASK_USABLE);
38 if (usable_a != usable_b) {
39 return (int) usable_b - (int) usable_a;
Marat Dukhan3045d4f2017-03-04 01:51:42 -050040 }
Marat Dukhan4f70b9a2017-09-25 23:18:41 -070041
42 /* Compare based on APIC ID (i.e. processor 0 < processor 1) */
43 const uint32_t id_a = processor_a->apic_id;
44 const uint32_t id_b = processor_b->apic_id;
45 return cmp(id_a, id_b);
Marat Dukhan3045d4f2017-03-04 01:51:42 -050046}
47
Marat Dukhan43a51cd2017-09-22 16:35:05 -070048static void cpuinfo_x86_count_objects(
Marat Dukhan4f70b9a2017-09-25 23:18:41 -070049 uint32_t linux_processors_count,
50 const struct cpuinfo_x86_linux_processor linux_processors[restrict static linux_processors_count],
51 const struct cpuinfo_x86_processor processor[restrict static 1],
Marat Dukhan43a51cd2017-09-22 16:35:05 -070052 uint32_t cores_count_ptr[restrict static 1],
53 uint32_t packages_count_ptr[restrict static 1],
Marat Dukhan3045d4f2017-03-04 01:51:42 -050054 uint32_t l1i_count_ptr[restrict static 1],
55 uint32_t l1d_count_ptr[restrict static 1],
56 uint32_t l2_count_ptr[restrict static 1],
57 uint32_t l3_count_ptr[restrict static 1],
58 uint32_t l4_count_ptr[restrict static 1])
59{
Marat Dukhan43a51cd2017-09-22 16:35:05 -070060 uint32_t cores_count = 0, packages_count = 0;
Marat Dukhan3045d4f2017-03-04 01:51:42 -050061 uint32_t l1i_count = 0, l1d_count = 0, l2_count = 0, l3_count = 0, l4_count = 0;
Marat Dukhan43a51cd2017-09-22 16:35:05 -070062 uint32_t last_core_id = UINT32_MAX, last_package_id = UINT32_MAX;
Marat Dukhan3045d4f2017-03-04 01:51:42 -050063 uint32_t last_l1i_id = UINT32_MAX, last_l1d_id = UINT32_MAX;
64 uint32_t last_l2_id = UINT32_MAX, last_l3_id = UINT32_MAX, last_l4_id = UINT32_MAX;
Marat Dukhan4f70b9a2017-09-25 23:18:41 -070065 for (uint32_t i = 0; i < linux_processors_count; i++) {
66 if (bitmask_all(linux_processors[i].flags, CPUINFO_LINUX_MASK_USABLE)) {
67 const uint32_t apic_id = linux_processors[i].apic_id;
68 cpuinfo_log_debug("APID ID %"PRIu32": system processor %"PRIu32, apic_id, linux_processors[i].linux_id);
69
70 /* All bits of APIC ID except thread ID mask */
71 const uint32_t core_id = apic_id &
72 ~(bit_mask(processor->topology.thread_bits_length) << processor->topology.thread_bits_offset);
73 if (core_id != last_core_id) {
74 last_core_id = core_id;
75 cores_count++;
Marat Dukhan3045d4f2017-03-04 01:51:42 -050076 }
Marat Dukhan4f70b9a2017-09-25 23:18:41 -070077 /* All bits of APIC ID except thread ID and core ID masks */
78 const uint32_t package_id = core_id &
79 ~(bit_mask(processor->topology.core_bits_length) << processor->topology.core_bits_offset);
80 if (package_id != last_package_id) {
81 last_package_id = package_id;
82 packages_count++;
Marat Dukhan3045d4f2017-03-04 01:51:42 -050083 }
Marat Dukhan4f70b9a2017-09-25 23:18:41 -070084 if (processor->cache.l1i.size != 0) {
85 const uint32_t l1i_id = apic_id & ~bit_mask(processor->cache.l1i.apic_bits);
86 if (l1i_id != last_l1i_id) {
87 last_l1i_id = l1i_id;
88 l1i_count++;
89 }
Marat Dukhan3045d4f2017-03-04 01:51:42 -050090 }
Marat Dukhan4f70b9a2017-09-25 23:18:41 -070091 if (processor->cache.l1d.size != 0) {
92 const uint32_t l1d_id = apic_id & ~bit_mask(processor->cache.l1d.apic_bits);
93 if (l1d_id != last_l1d_id) {
94 last_l1d_id = l1d_id;
95 l1d_count++;
96 }
Marat Dukhan3045d4f2017-03-04 01:51:42 -050097 }
Marat Dukhan4f70b9a2017-09-25 23:18:41 -070098 if (processor->cache.l2.size != 0) {
99 const uint32_t l2_id = apic_id & ~bit_mask(processor->cache.l2.apic_bits);
100 if (l2_id != last_l2_id) {
101 last_l2_id = l2_id;
102 l2_count++;
103 }
104 }
105 if (processor->cache.l3.size != 0) {
106 const uint32_t l3_id = apic_id & ~bit_mask(processor->cache.l3.apic_bits);
107 if (l3_id != last_l3_id) {
108 last_l3_id = l3_id;
109 l3_count++;
110 }
111 }
112 if (processor->cache.l4.size != 0) {
113 const uint32_t l4_id = apic_id & ~bit_mask(processor->cache.l4.apic_bits);
114 if (l4_id != last_l4_id) {
115 last_l4_id = l4_id;
116 l4_count++;
117 }
Marat Dukhan3045d4f2017-03-04 01:51:42 -0500118 }
119 }
120 }
Marat Dukhan43a51cd2017-09-22 16:35:05 -0700121 *cores_count_ptr = cores_count;
122 *packages_count_ptr = packages_count;
Marat Dukhan3045d4f2017-03-04 01:51:42 -0500123 *l1i_count_ptr = l1i_count;
124 *l1d_count_ptr = l1d_count;
125 *l2_count_ptr = l2_count;
126 *l3_count_ptr = l3_count;
127 *l4_count_ptr = l4_count;
128}
129
130void cpuinfo_x86_linux_init(void) {
Marat Dukhan4f70b9a2017-09-25 23:18:41 -0700131 struct cpuinfo_x86_linux_processor* x86_linux_processors = NULL;
132 struct cpuinfo_processor* processors = NULL;
133 struct cpuinfo_core* cores = NULL;
134 struct cpuinfo_package* packages = NULL;
Marat Dukhand9f76e02017-09-24 22:46:08 -0700135 const struct cpuinfo_processor** linux_cpu_to_processor_map = NULL;
136 const struct cpuinfo_core** linux_cpu_to_core_map = NULL;
Marat Dukhan3045d4f2017-03-04 01:51:42 -0500137 struct cpuinfo_cache* l1i = NULL;
138 struct cpuinfo_cache* l1d = NULL;
139 struct cpuinfo_cache* l2 = NULL;
140 struct cpuinfo_cache* l3 = NULL;
141 struct cpuinfo_cache* l4 = NULL;
142
Marat Dukhan4f70b9a2017-09-25 23:18:41 -0700143 const uint32_t max_processors_count = cpuinfo_linux_get_max_processors_count();
144 cpuinfo_log_debug("system maximum processors count: %"PRIu32, max_processors_count);
145
146 const uint32_t max_possible_processors_count = 1 +
147 cpuinfo_linux_get_max_possible_processor(max_processors_count);
148 cpuinfo_log_debug("maximum possible processors count: %"PRIu32, max_possible_processors_count);
149 const uint32_t max_present_processors_count = 1 +
Marat Dukhan72c3c382018-01-08 17:33:40 -0800150 cpuinfo_linux_get_max_present_processor(max_processors_count);
Marat Dukhan4f70b9a2017-09-25 23:18:41 -0700151 cpuinfo_log_debug("maximum present processors count: %"PRIu32, max_present_processors_count);
152
153 const uint32_t x86_linux_processors_count = min(max_possible_processors_count, max_present_processors_count);
154 x86_linux_processors = calloc(x86_linux_processors_count, sizeof(struct cpuinfo_x86_linux_processor));
155 if (x86_linux_processors == NULL) {
156 cpuinfo_log_error(
157 "failed to allocate %zu bytes for descriptions of %"PRIu32" x86 logical processors",
158 x86_linux_processors_count * sizeof(struct cpuinfo_x86_linux_processor),
159 x86_linux_processors_count);
Marat Dukhan3045d4f2017-03-04 01:51:42 -0500160 return;
161 }
162
Marat Dukhan4f70b9a2017-09-25 23:18:41 -0700163 cpuinfo_linux_detect_possible_processors(
164 x86_linux_processors_count, &x86_linux_processors->flags,
165 sizeof(struct cpuinfo_x86_linux_processor),
166 CPUINFO_LINUX_FLAG_POSSIBLE);
Marat Dukhan3045d4f2017-03-04 01:51:42 -0500167
Marat Dukhan4f70b9a2017-09-25 23:18:41 -0700168 cpuinfo_linux_detect_present_processors(
169 x86_linux_processors_count, &x86_linux_processors->flags,
170 sizeof(struct cpuinfo_x86_linux_processor),
171 CPUINFO_LINUX_FLAG_PRESENT);
Marat Dukhan3045d4f2017-03-04 01:51:42 -0500172
Marat Dukhan4f70b9a2017-09-25 23:18:41 -0700173 if (!cpuinfo_x86_linux_parse_proc_cpuinfo(x86_linux_processors_count, x86_linux_processors)) {
174 cpuinfo_log_error("failed to parse processor information from /proc/cpuinfo");
175 return;
Marat Dukhan3045d4f2017-03-04 01:51:42 -0500176 }
177
Marat Dukhan4f70b9a2017-09-25 23:18:41 -0700178 struct cpuinfo_x86_processor x86_processor;
179 memset(&x86_processor, 0, sizeof(x86_processor));
180 cpuinfo_x86_init_processor(&x86_processor);
Marat Dukhan88346572018-03-11 15:28:17 -0700181 char brand_string[48];
182 cpuinfo_x86_normalize_brand_string(x86_processor.brand_string, brand_string);
Marat Dukhan3045d4f2017-03-04 01:51:42 -0500183
Marat Dukhan4f70b9a2017-09-25 23:18:41 -0700184 uint32_t processors_count = 0;
185 for (uint32_t i = 0; i < x86_linux_processors_count; i++) {
186 if (bitmask_all(x86_linux_processors[i].flags, CPUINFO_LINUX_MASK_USABLE)) {
187 x86_linux_processors[i].linux_id = i;
188 processors_count++;
189 }
Marat Dukhan3045d4f2017-03-04 01:51:42 -0500190 }
191
Marat Dukhan4f70b9a2017-09-25 23:18:41 -0700192 qsort(x86_linux_processors, x86_linux_processors_count, sizeof(struct cpuinfo_x86_linux_processor),
193 cmp_x86_linux_processor);
Marat Dukhan3045d4f2017-03-04 01:51:42 -0500194
Marat Dukhan4f70b9a2017-09-25 23:18:41 -0700195 processors = calloc(processors_count, sizeof(struct cpuinfo_processor));
Marat Dukhan3045d4f2017-03-04 01:51:42 -0500196 if (processors == NULL) {
Marat Dukhan4f70b9a2017-09-25 23:18:41 -0700197 cpuinfo_log_error("failed to allocate %zu bytes for descriptions of %"PRIu32" logical processors",
198 processors_count * sizeof(struct cpuinfo_processor), processors_count);
Marat Dukhan3045d4f2017-03-04 01:51:42 -0500199 goto cleanup;
200 }
201
Marat Dukhan43a51cd2017-09-22 16:35:05 -0700202 uint32_t packages_count = 0, cores_count = 0;
Marat Dukhan3045d4f2017-03-04 01:51:42 -0500203 uint32_t l1i_count = 0, l1d_count = 0, l2_count = 0, l3_count = 0, l4_count = 0;
Marat Dukhan4f70b9a2017-09-25 23:18:41 -0700204 cpuinfo_x86_count_objects(x86_linux_processors_count, x86_linux_processors, &x86_processor,
Marat Dukhan38dc29a2017-09-22 17:33:25 -0700205 &cores_count, &packages_count, &l1i_count, &l1d_count, &l2_count, &l3_count, &l4_count);
Marat Dukhan3045d4f2017-03-04 01:51:42 -0500206
Marat Dukhan4f70b9a2017-09-25 23:18:41 -0700207 cpuinfo_log_debug("detected %"PRIu32" cores", cores_count);
208 cpuinfo_log_debug("detected %"PRIu32" packages", packages_count);
Marat Dukhan43a51cd2017-09-22 16:35:05 -0700209 cpuinfo_log_debug("detected %"PRIu32" L1I caches", l1i_count);
210 cpuinfo_log_debug("detected %"PRIu32" L1D caches", l1d_count);
211 cpuinfo_log_debug("detected %"PRIu32" L2 caches", l2_count);
212 cpuinfo_log_debug("detected %"PRIu32" L3 caches", l3_count);
213 cpuinfo_log_debug("detected %"PRIu32" L4 caches", l4_count);
Marat Dukhan3045d4f2017-03-04 01:51:42 -0500214
Marat Dukhan02efbc92017-10-28 14:19:18 -0400215 linux_cpu_to_processor_map = calloc(x86_linux_processors_count, sizeof(struct cpuinfo_processor*));
Marat Dukhand9f76e02017-09-24 22:46:08 -0700216 if (linux_cpu_to_processor_map == NULL) {
217 cpuinfo_log_error("failed to allocate %zu bytes for mapping entries of %"PRIu32" logical processors",
Marat Dukhan02efbc92017-10-28 14:19:18 -0400218 x86_linux_processors_count * sizeof(struct cpuinfo_processor*),
219 x86_linux_processors_count);
Marat Dukhand9f76e02017-09-24 22:46:08 -0700220 goto cleanup;
221 }
222
Marat Dukhan02efbc92017-10-28 14:19:18 -0400223 linux_cpu_to_core_map = calloc(x86_linux_processors_count, sizeof(struct cpuinfo_core*));
Marat Dukhand9f76e02017-09-24 22:46:08 -0700224 if (linux_cpu_to_core_map == NULL) {
225 cpuinfo_log_error("failed to allocate %zu bytes for mapping entries of %"PRIu32" cores",
Marat Dukhan02efbc92017-10-28 14:19:18 -0400226 x86_linux_processors_count * sizeof(struct cpuinfo_core*),
227 x86_linux_processors_count);
Marat Dukhand9f76e02017-09-24 22:46:08 -0700228 goto cleanup;
229 }
230
Marat Dukhan43a51cd2017-09-22 16:35:05 -0700231 cores = calloc(cores_count, sizeof(struct cpuinfo_core));
232 if (cores == NULL) {
233 cpuinfo_log_error("failed to allocate %zu bytes for descriptions of %"PRIu32" cores",
234 cores_count * sizeof(struct cpuinfo_core), cores_count);
235 goto cleanup;
236 }
237 packages = calloc(packages_count, sizeof(struct cpuinfo_package));
238 if (packages == NULL) {
Marat Dukhan29000602018-03-18 00:29:39 -0700239 cpuinfo_log_error("failed to allocate %zu bytes for descriptions of %"PRIu32" physical packages",
Marat Dukhan43a51cd2017-09-22 16:35:05 -0700240 packages_count * sizeof(struct cpuinfo_package), packages_count);
241 goto cleanup;
242 }
Marat Dukhan29000602018-03-18 00:29:39 -0700243
Marat Dukhan3045d4f2017-03-04 01:51:42 -0500244 if (l1i_count != 0) {
245 l1i = calloc(l1i_count, sizeof(struct cpuinfo_cache));
246 if (l1i == NULL) {
247 cpuinfo_log_error("failed to allocate %zu bytes for descriptions of %"PRIu32" L1I caches",
248 l1i_count * sizeof(struct cpuinfo_cache), l1i_count);
249 goto cleanup;
250 }
251 }
252 if (l1d_count != 0) {
253 l1d = calloc(l1d_count, sizeof(struct cpuinfo_cache));
254 if (l1d == NULL) {
255 cpuinfo_log_error("failed to allocate %zu bytes for descriptions of %"PRIu32" L1D caches",
256 l1d_count * sizeof(struct cpuinfo_cache), l1d_count);
257 goto cleanup;
258 }
259 }
260 if (l2_count != 0) {
261 l2 = calloc(l2_count, sizeof(struct cpuinfo_cache));
262 if (l2 == NULL) {
263 cpuinfo_log_error("failed to allocate %zu bytes for descriptions of %"PRIu32" L2 caches",
264 l2_count * sizeof(struct cpuinfo_cache), l2_count);
265 goto cleanup;
266 }
267 }
268 if (l3_count != 0) {
269 l3 = calloc(l3_count, sizeof(struct cpuinfo_cache));
270 if (l3 == NULL) {
271 cpuinfo_log_error("failed to allocate %zu bytes for descriptions of %"PRIu32" L3 caches",
272 l3_count * sizeof(struct cpuinfo_cache), l3_count);
273 goto cleanup;
274 }
275 }
276 if (l4_count != 0) {
277 l4 = calloc(l4_count, sizeof(struct cpuinfo_cache));
278 if (l4 == NULL) {
279 cpuinfo_log_error("failed to allocate %zu bytes for descriptions of %"PRIu32" L4 caches",
280 l4_count * sizeof(struct cpuinfo_cache), l4_count);
281 goto cleanup;
282 }
283 }
284
Marat Dukhan4f70b9a2017-09-25 23:18:41 -0700285 uint32_t processor_index = UINT32_MAX, core_index = UINT32_MAX, package_index = UINT32_MAX;
Marat Dukhanfbc4cfc2017-09-22 16:45:45 -0700286 uint32_t l1i_index = UINT32_MAX, l1d_index = UINT32_MAX, l2_index = UINT32_MAX, l3_index = UINT32_MAX, l4_index = UINT32_MAX;
Marat Dukhan4f70b9a2017-09-25 23:18:41 -0700287 uint32_t core_id = 0, smt_id = 0;
288 uint32_t last_apic_core_id = UINT32_MAX, last_apic_package_id = UINT32_MAX;
Marat Dukhan3045d4f2017-03-04 01:51:42 -0500289 uint32_t last_l1i_id = UINT32_MAX, last_l1d_id = UINT32_MAX;
290 uint32_t last_l2_id = UINT32_MAX, last_l3_id = UINT32_MAX, last_l4_id = UINT32_MAX;
Marat Dukhan4f70b9a2017-09-25 23:18:41 -0700291 for (uint32_t i = 0; i < x86_linux_processors_count; i++) {
292 if (bitmask_all(x86_linux_processors[i].flags, CPUINFO_LINUX_MASK_USABLE)) {
293 const uint32_t apic_id = x86_linux_processors[i].apic_id;
294 processor_index++;
295 smt_id++;
Marat Dukhan38dc29a2017-09-22 17:33:25 -0700296
Marat Dukhan4f70b9a2017-09-25 23:18:41 -0700297 /* All bits of APIC ID except thread ID mask */
298 const uint32_t apid_core_id = apic_id &
299 ~(bit_mask(x86_processor.topology.thread_bits_length) << x86_processor.topology.thread_bits_offset);
300 if (apid_core_id != last_apic_core_id) {
301 core_index++;
302 core_id++;
303 smt_id = 0;
Marat Dukhan43a51cd2017-09-22 16:35:05 -0700304 }
Marat Dukhan4f70b9a2017-09-25 23:18:41 -0700305 /* All bits of APIC ID except thread ID and core ID masks */
306 const uint32_t apic_package_id = apid_core_id &
307 ~(bit_mask(x86_processor.topology.core_bits_length) << x86_processor.topology.core_bits_offset);
308 if (apic_package_id != last_apic_package_id) {
309 package_index++;
310 core_id = 0;
Marat Dukhan3045d4f2017-03-04 01:51:42 -0500311 }
Marat Dukhan4f70b9a2017-09-25 23:18:41 -0700312
313 /* Initialize logical processor object */
314 processors[processor_index].smt_id = smt_id;
315 processors[processor_index].core = cores + core_index;
316 processors[processor_index].package = packages + package_index;
317 processors[processor_index].linux_id = x86_linux_processors[i].linux_id;
318 processors[processor_index].apic_id = x86_linux_processors[i].apic_id;
319
320 if (apid_core_id != last_apic_core_id) {
321 /* new core */
322 cores[core_index] = (struct cpuinfo_core) {
323 .processor_start = processor_index,
Marat Dukhanab3a1272017-08-25 23:20:07 -0700324 .processor_count = 1,
Marat Dukhan4f70b9a2017-09-25 23:18:41 -0700325 .core_id = core_id,
326 .package = packages + package_index,
327 .vendor = x86_processor.vendor,
328 .uarch = x86_processor.uarch,
329 .cpuid = x86_processor.cpuid,
Marat Dukhan3045d4f2017-03-04 01:51:42 -0500330 };
Marat Dukhan4f70b9a2017-09-25 23:18:41 -0700331 packages[package_index].core_count += 1;
332 last_apic_core_id = apid_core_id;
Marat Dukhan3045d4f2017-03-04 01:51:42 -0500333 } else {
Marat Dukhan4f70b9a2017-09-25 23:18:41 -0700334 /* another logical processor on the same core */
335 cores[core_index].processor_count++;
Marat Dukhan3045d4f2017-03-04 01:51:42 -0500336 }
Marat Dukhan4f70b9a2017-09-25 23:18:41 -0700337
338 if (apic_package_id != last_apic_package_id) {
339 /* new package */
340 packages[package_index].processor_start = processor_index;
341 packages[package_index].processor_count = 1;
342 packages[package_index].core_start = core_index;
Marat Dukhan88346572018-03-11 15:28:17 -0700343 cpuinfo_x86_format_package_name(x86_processor.vendor, brand_string, packages[package_index].name);
Marat Dukhan4f70b9a2017-09-25 23:18:41 -0700344 last_apic_package_id = apic_package_id;
Marat Dukhan3045d4f2017-03-04 01:51:42 -0500345 } else {
Marat Dukhan4f70b9a2017-09-25 23:18:41 -0700346 /* another logical processor on the same package */
347 packages[package_index].processor_count++;
Marat Dukhan3045d4f2017-03-04 01:51:42 -0500348 }
Marat Dukhan4f70b9a2017-09-25 23:18:41 -0700349
350 linux_cpu_to_processor_map[x86_linux_processors[i].linux_id] = processors + processor_index;
351 linux_cpu_to_core_map[x86_linux_processors[i].linux_id] = cores + core_index;
352
353 if (x86_processor.cache.l1i.size != 0) {
354 const uint32_t l1i_id = apic_id & ~bit_mask(x86_processor.cache.l1i.apic_bits);
355 processors[i].cache.l1i = &l1i[l1i_index];
356 if (l1i_id != last_l1i_id) {
357 /* new cache */
358 last_l1i_id = l1i_id;
359 l1i[++l1i_index] = (struct cpuinfo_cache) {
360 .size = x86_processor.cache.l1i.size,
361 .associativity = x86_processor.cache.l1i.associativity,
362 .sets = x86_processor.cache.l1i.sets,
363 .partitions = x86_processor.cache.l1i.partitions,
364 .line_size = x86_processor.cache.l1i.line_size,
365 .flags = x86_processor.cache.l1i.flags,
366 .processor_start = processor_index,
367 .processor_count = 1,
368 };
369 } else {
370 /* another processor sharing the same cache */
371 l1i[l1i_index].processor_count += 1;
372 }
373 processors[i].cache.l1i = &l1i[l1i_index];
Marat Dukhan3045d4f2017-03-04 01:51:42 -0500374 } else {
Marat Dukhan4f70b9a2017-09-25 23:18:41 -0700375 /* reset cache id */
376 last_l1i_id = UINT32_MAX;
Marat Dukhan3045d4f2017-03-04 01:51:42 -0500377 }
Marat Dukhan4f70b9a2017-09-25 23:18:41 -0700378 if (x86_processor.cache.l1d.size != 0) {
379 const uint32_t l1d_id = apic_id & ~bit_mask(x86_processor.cache.l1d.apic_bits);
380 processors[i].cache.l1d = &l1d[l1d_index];
381 if (l1d_id != last_l1d_id) {
382 /* new cache */
383 last_l1d_id = l1d_id;
384 l1d[++l1d_index] = (struct cpuinfo_cache) {
385 .size = x86_processor.cache.l1d.size,
386 .associativity = x86_processor.cache.l1d.associativity,
387 .sets = x86_processor.cache.l1d.sets,
388 .partitions = x86_processor.cache.l1d.partitions,
389 .line_size = x86_processor.cache.l1d.line_size,
390 .flags = x86_processor.cache.l1d.flags,
391 .processor_start = processor_index,
392 .processor_count = 1,
393 };
394 } else {
395 /* another processor sharing the same cache */
396 l1d[l1d_index].processor_count += 1;
397 }
398 processors[i].cache.l1d = &l1d[l1d_index];
399 } else {
400 /* reset cache id */
401 last_l1d_id = UINT32_MAX;
402 }
403 if (x86_processor.cache.l2.size != 0) {
404 const uint32_t l2_id = apic_id & ~bit_mask(x86_processor.cache.l2.apic_bits);
405 processors[i].cache.l2 = &l2[l2_index];
406 if (l2_id != last_l2_id) {
407 /* new cache */
408 last_l2_id = l2_id;
409 l2[++l2_index] = (struct cpuinfo_cache) {
410 .size = x86_processor.cache.l2.size,
411 .associativity = x86_processor.cache.l2.associativity,
412 .sets = x86_processor.cache.l2.sets,
413 .partitions = x86_processor.cache.l2.partitions,
414 .line_size = x86_processor.cache.l2.line_size,
415 .flags = x86_processor.cache.l2.flags,
416 .processor_start = processor_index,
417 .processor_count = 1,
418 };
419 } else {
420 /* another processor sharing the same cache */
421 l2[l2_index].processor_count += 1;
422 }
423 processors[i].cache.l2 = &l2[l2_index];
424 } else {
425 /* reset cache id */
426 last_l2_id = UINT32_MAX;
427 }
428 if (x86_processor.cache.l3.size != 0) {
429 const uint32_t l3_id = apic_id & ~bit_mask(x86_processor.cache.l3.apic_bits);
430 processors[i].cache.l3 = &l3[l3_index];
431 if (l3_id != last_l3_id) {
432 /* new cache */
433 last_l3_id = l3_id;
434 l3[++l3_index] = (struct cpuinfo_cache) {
435 .size = x86_processor.cache.l3.size,
436 .associativity = x86_processor.cache.l3.associativity,
437 .sets = x86_processor.cache.l3.sets,
438 .partitions = x86_processor.cache.l3.partitions,
439 .line_size = x86_processor.cache.l3.line_size,
440 .flags = x86_processor.cache.l3.flags,
441 .processor_start = processor_index,
442 .processor_count = 1,
443 };
444 } else {
445 /* another processor sharing the same cache */
446 l3[l3_index].processor_count += 1;
447 }
448 processors[i].cache.l3 = &l3[l3_index];
449 } else {
450 /* reset cache id */
451 last_l3_id = UINT32_MAX;
452 }
453 if (x86_processor.cache.l4.size != 0) {
454 const uint32_t l4_id = apic_id & ~bit_mask(x86_processor.cache.l4.apic_bits);
455 processors[i].cache.l4 = &l4[l4_index];
456 if (l4_id != last_l4_id) {
457 /* new cache */
458 last_l4_id = l4_id;
459 l4[++l4_index] = (struct cpuinfo_cache) {
460 .size = x86_processor.cache.l4.size,
461 .associativity = x86_processor.cache.l4.associativity,
462 .sets = x86_processor.cache.l4.sets,
463 .partitions = x86_processor.cache.l4.partitions,
464 .line_size = x86_processor.cache.l4.line_size,
465 .flags = x86_processor.cache.l4.flags,
466 .processor_start = processor_index,
467 .processor_count = 1,
468 };
469 } else {
470 /* another processor sharing the same cache */
471 l4[l4_index].processor_count += 1;
472 }
473 processors[i].cache.l4 = &l4[l4_index];
474 } else {
475 /* reset cache id */
476 last_l4_id = UINT32_MAX;
477 }
Marat Dukhan3045d4f2017-03-04 01:51:42 -0500478 }
479 }
480
Marat Dukhanfbc4cfc2017-09-22 16:45:45 -0700481 #ifdef __ANDROID__
482 cpuinfo_gpu_query_gles2(packages[0].gpu_name);
Marat Dukhanfd0f3ef2017-12-18 17:45:18 -0800483 struct cpuinfo_android_gpu gpu = cpuinfo_android_decode_gpu(packages[0].gpu_name);
484 if (gpu.series != cpuinfo_android_gpu_series_unknown) {
485 cpuinfo_android_gpu_to_string(&gpu, packages[0].gpu_name);
486 }
Marat Dukhanfbc4cfc2017-09-22 16:45:45 -0700487 #endif
488
Marat Dukhan3045d4f2017-03-04 01:51:42 -0500489 /* Commit changes */
Marat Dukhand9f76e02017-09-24 22:46:08 -0700490 cpuinfo_linux_cpu_to_processor_map = linux_cpu_to_processor_map;
491 cpuinfo_linux_cpu_to_core_map = linux_cpu_to_core_map;
492
Marat Dukhan3045d4f2017-03-04 01:51:42 -0500493 cpuinfo_processors = processors;
Marat Dukhan43a51cd2017-09-22 16:35:05 -0700494 cpuinfo_cores = cores;
495 cpuinfo_packages = packages;
Marat Dukhan3045d4f2017-03-04 01:51:42 -0500496 cpuinfo_cache[cpuinfo_cache_level_1i] = l1i;
497 cpuinfo_cache[cpuinfo_cache_level_1d] = l1d;
498 cpuinfo_cache[cpuinfo_cache_level_2] = l2;
499 cpuinfo_cache[cpuinfo_cache_level_3] = l3;
500 cpuinfo_cache[cpuinfo_cache_level_4] = l4;
501
Marat Dukhan8ecad1a2017-05-08 07:21:57 +0000502 cpuinfo_processors_count = processors_count;
Marat Dukhan43a51cd2017-09-22 16:35:05 -0700503 cpuinfo_cores_count = cores_count;
504 cpuinfo_packages_count = packages_count;
Marat Dukhan3045d4f2017-03-04 01:51:42 -0500505 cpuinfo_cache_count[cpuinfo_cache_level_1i] = l1i_count;
506 cpuinfo_cache_count[cpuinfo_cache_level_1d] = l1d_count;
507 cpuinfo_cache_count[cpuinfo_cache_level_2] = l2_count;
508 cpuinfo_cache_count[cpuinfo_cache_level_3] = l3_count;
509 cpuinfo_cache_count[cpuinfo_cache_level_4] = l4_count;
510
Marat Dukhand9f76e02017-09-24 22:46:08 -0700511 linux_cpu_to_processor_map = NULL;
512 linux_cpu_to_core_map = NULL;
Marat Dukhan3045d4f2017-03-04 01:51:42 -0500513 processors = NULL;
Marat Dukhan43a51cd2017-09-22 16:35:05 -0700514 cores = NULL;
515 packages = NULL;
Marat Dukhan3045d4f2017-03-04 01:51:42 -0500516 l1i = l1d = l2 = l3 = l4 = NULL;
517
518cleanup:
Marat Dukhand9f76e02017-09-24 22:46:08 -0700519 free(linux_cpu_to_processor_map);
520 free(linux_cpu_to_core_map);
Marat Dukhan4f70b9a2017-09-25 23:18:41 -0700521 free(x86_linux_processors);
Marat Dukhan3045d4f2017-03-04 01:51:42 -0500522 free(processors);
Marat Dukhan43a51cd2017-09-22 16:35:05 -0700523 free(cores);
524 free(packages);
Marat Dukhan3045d4f2017-03-04 01:51:42 -0500525 free(l1i);
526 free(l1d);
527 free(l2);
528 free(l3);
529 free(l4);
530}