gpio: rcar: Fix runtime PM imbalance on error
[linux/fpc-iii.git] / tools / perf / util / affinity.c
bloba5e31f82682804c3f6c837bc82e4197a6e649e4e
1 // SPDX-License-Identifier: GPL-2.0
2 /* Manage affinity to optimize IPIs inside the kernel perf API. */
3 #define _GNU_SOURCE 1
4 #include <sched.h>
5 #include <stdlib.h>
6 #include <linux/bitmap.h>
7 #include <linux/zalloc.h>
8 #include "perf.h"
9 #include "cpumap.h"
10 #include "affinity.h"
12 static int get_cpu_set_size(void)
14 int sz = cpu__max_cpu() + 8 - 1;
16 * sched_getaffinity doesn't like masks smaller than the kernel.
17 * Hopefully that's big enough.
19 if (sz < 4096)
20 sz = 4096;
21 return sz / 8;
24 int affinity__setup(struct affinity *a)
26 int cpu_set_size = get_cpu_set_size();
28 a->orig_cpus = bitmap_alloc(cpu_set_size * 8);
29 if (!a->orig_cpus)
30 return -1;
31 sched_getaffinity(0, cpu_set_size, (cpu_set_t *)a->orig_cpus);
32 a->sched_cpus = bitmap_alloc(cpu_set_size * 8);
33 if (!a->sched_cpus) {
34 zfree(&a->orig_cpus);
35 return -1;
37 bitmap_zero((unsigned long *)a->sched_cpus, cpu_set_size);
38 a->changed = false;
39 return 0;
43 * perf_event_open does an IPI internally to the target CPU.
44 * It is more efficient to change perf's affinity to the target
45 * CPU and then set up all events on that CPU, so we amortize
46 * CPU communication.
48 void affinity__set(struct affinity *a, int cpu)
50 int cpu_set_size = get_cpu_set_size();
52 if (cpu == -1)
53 return;
54 a->changed = true;
55 set_bit(cpu, a->sched_cpus);
57 * We ignore errors because affinity is just an optimization.
58 * This could happen for example with isolated CPUs or cpusets.
59 * In this case the IPIs inside the kernel's perf API still work.
61 sched_setaffinity(0, cpu_set_size, (cpu_set_t *)a->sched_cpus);
62 clear_bit(cpu, a->sched_cpus);
65 void affinity__cleanup(struct affinity *a)
67 int cpu_set_size = get_cpu_set_size();
69 if (a->changed)
70 sched_setaffinity(0, cpu_set_size, (cpu_set_t *)a->orig_cpus);
71 zfree(&a->sched_cpus);
72 zfree(&a->orig_cpus);