]> xenbits.xensource.com Git - people/aperard/linux.git/commitdiff
selftests/seccomp: Pin benchmark to single CPU
authorKees Cook <keescook@chromium.org>
Tue, 6 Feb 2024 09:53:19 +0000 (01:53 -0800)
committerKees Cook <keescook@chromium.org>
Wed, 7 Feb 2024 11:10:57 +0000 (03:10 -0800)
The seccomp benchmark test (for validating the benefit of bitmaps) can
be sensitive to scheduling speed, so pin the process to a single CPU,
which appears to significantly improve reliability, and loosen the
"close enough" checking to allow up to 10% variance instead of 1%.

Reported-by: kernel test robot <oliver.sang@intel.com>
Closes: https://lore.kernel.org/oe-lkp/202402061002.3a8722fd-oliver.sang@intel.com
Cc: Andy Lutomirski <luto@amacapital.net>
Cc: Will Drewry <wad@chromium.org>
Reviewed-by: Mark Brown <broonie@kernel.org>
Signed-off-by: Kees Cook <keescook@chromium.org>
tools/testing/selftests/seccomp/seccomp_benchmark.c

index 5b5c9d558dee07bc1f7afd7df280e1189858451e..9d7aa5a730e017f24b4ce2fcdae40b923212116a 100644 (file)
@@ -4,7 +4,9 @@
  */
 #define _GNU_SOURCE
 #include <assert.h>
+#include <err.h>
 #include <limits.h>
+#include <sched.h>
 #include <stdbool.h>
 #include <stddef.h>
 #include <stdio.h>
@@ -76,8 +78,12 @@ unsigned long long calibrate(void)
 
 bool approx(int i_one, int i_two)
 {
-       double one = i_one, one_bump = one * 0.01;
-       double two = i_two, two_bump = two * 0.01;
+       /*
+        * This continues to be a noisy test. Instead of a 1% comparison
+        * go with 10%.
+        */
+       double one = i_one, one_bump = one * 0.1;
+       double two = i_two, two_bump = two * 0.1;
 
        one_bump = one + MAX(one_bump, 2.0);
        two_bump = two + MAX(two_bump, 2.0);
@@ -119,6 +125,32 @@ long compare(const char *name_one, const char *name_eval, const char *name_two,
        return good ? 0 : 1;
 }
 
+/* Pin to a single CPU so the benchmark won't bounce around the system. */
+void affinity(void)
+{
+       long cpu;
+       ulong ncores = sysconf(_SC_NPROCESSORS_CONF);
+       cpu_set_t *setp = CPU_ALLOC(ncores);
+       ulong setsz = CPU_ALLOC_SIZE(ncores);
+
+       /*
+        * Totally unscientific way to avoid CPUs that might be busier:
+        * choose the highest CPU instead of the lowest.
+        */
+       for (cpu = ncores - 1; cpu >= 0; cpu--) {
+               CPU_ZERO_S(setsz, setp);
+               CPU_SET_S(cpu, setsz, setp);
+               if (sched_setaffinity(getpid(), setsz, setp) == -1)
+                       continue;
+               printf("Pinned to CPU %lu of %lu\n", cpu + 1, ncores);
+               goto out;
+       }
+       fprintf(stderr, "Could not set CPU affinity -- calibration may not work well");
+
+out:
+       CPU_FREE(setp);
+}
+
 int main(int argc, char *argv[])
 {
        struct sock_filter bitmap_filter[] = {
@@ -153,6 +185,8 @@ int main(int argc, char *argv[])
        system("grep -H . /proc/sys/net/core/bpf_jit_enable");
        system("grep -H . /proc/sys/net/core/bpf_jit_harden");
 
+       affinity();
+
        if (argc > 1)
                samples = strtoull(argv[1], NULL, 0);
        else