3a72c94ebf
The rfi_flush and entry_flush selftests work by using the PM_LD_MISS_L1 perf event to count L1D misses. The value of this event has changed over time: - Power7 uses 0x400f0 - Power8 and Power9 use both 0x400f0 and 0x3e054 - Power10 uses only 0x3e054 Rather than relying on raw values, configure perf to count L1D read misses in the most explicit way available. This fixes the selftests to work on systems without 0x400f0 as PM_LD_MISS_L1, and should change no behaviour for systems that the tests already worked on. The only potential downside is that referring to a specific perf event requires PMU support implemented in the kernel for that platform. Signed-off-by: Russell Currey <ruscur@russell.cc> Acked-by: Daniel Axtens <dja@axtens.net> Signed-off-by: Michael Ellerman <mpe@ellerman.id.au> Link: https://lore.kernel.org/r/20210223070227.2916871-1-ruscur@russell.cc
140 lines
3.4 KiB
C
140 lines
3.4 KiB
C
// SPDX-License-Identifier: GPL-2.0+
|
|
|
|
/*
|
|
* Copyright 2018 IBM Corporation.
|
|
*/
|
|
|
|
#define __SANE_USERSPACE_TYPES__
|
|
|
|
#include <sys/types.h>
|
|
#include <stdint.h>
|
|
#include <malloc.h>
|
|
#include <unistd.h>
|
|
#include <signal.h>
|
|
#include <stdlib.h>
|
|
#include <string.h>
|
|
#include <stdio.h>
|
|
#include "utils.h"
|
|
#include "flush_utils.h"
|
|
|
|
int entry_flush_test(void)
|
|
{
|
|
char *p;
|
|
int repetitions = 10;
|
|
int fd, passes = 0, iter, rc = 0;
|
|
struct perf_event_read v;
|
|
__u64 l1d_misses_total = 0;
|
|
unsigned long iterations = 100000, zero_size = 24 * 1024;
|
|
unsigned long l1d_misses_expected;
|
|
int rfi_flush_orig;
|
|
int entry_flush, entry_flush_orig;
|
|
|
|
SKIP_IF(geteuid() != 0);
|
|
|
|
// The PMU event we use only works on Power7 or later
|
|
SKIP_IF(!have_hwcap(PPC_FEATURE_ARCH_2_06));
|
|
|
|
if (read_debugfs_file("powerpc/rfi_flush", &rfi_flush_orig) < 0) {
|
|
perror("Unable to read powerpc/rfi_flush debugfs file");
|
|
SKIP_IF(1);
|
|
}
|
|
|
|
if (read_debugfs_file("powerpc/entry_flush", &entry_flush_orig) < 0) {
|
|
perror("Unable to read powerpc/entry_flush debugfs file");
|
|
SKIP_IF(1);
|
|
}
|
|
|
|
if (rfi_flush_orig != 0) {
|
|
if (write_debugfs_file("powerpc/rfi_flush", 0) < 0) {
|
|
perror("error writing to powerpc/rfi_flush debugfs file");
|
|
FAIL_IF(1);
|
|
}
|
|
}
|
|
|
|
entry_flush = entry_flush_orig;
|
|
|
|
fd = perf_event_open_counter(PERF_TYPE_HW_CACHE, PERF_L1D_READ_MISS_CONFIG, -1);
|
|
FAIL_IF(fd < 0);
|
|
|
|
p = (char *)memalign(zero_size, CACHELINE_SIZE);
|
|
|
|
FAIL_IF(perf_event_enable(fd));
|
|
|
|
// disable L1 prefetching
|
|
set_dscr(1);
|
|
|
|
iter = repetitions;
|
|
|
|
/*
|
|
* We expect to see l1d miss for each cacheline access when entry_flush
|
|
* is set. Allow a small variation on this.
|
|
*/
|
|
l1d_misses_expected = iterations * (zero_size / CACHELINE_SIZE - 2);
|
|
|
|
again:
|
|
FAIL_IF(perf_event_reset(fd));
|
|
|
|
syscall_loop(p, iterations, zero_size);
|
|
|
|
FAIL_IF(read(fd, &v, sizeof(v)) != sizeof(v));
|
|
|
|
if (entry_flush && v.l1d_misses >= l1d_misses_expected)
|
|
passes++;
|
|
else if (!entry_flush && v.l1d_misses < (l1d_misses_expected / 2))
|
|
passes++;
|
|
|
|
l1d_misses_total += v.l1d_misses;
|
|
|
|
while (--iter)
|
|
goto again;
|
|
|
|
if (passes < repetitions) {
|
|
printf("FAIL (L1D misses with entry_flush=%d: %llu %c %lu) [%d/%d failures]\n",
|
|
entry_flush, l1d_misses_total, entry_flush ? '<' : '>',
|
|
entry_flush ? repetitions * l1d_misses_expected :
|
|
repetitions * l1d_misses_expected / 2,
|
|
repetitions - passes, repetitions);
|
|
rc = 1;
|
|
} else {
|
|
printf("PASS (L1D misses with entry_flush=%d: %llu %c %lu) [%d/%d pass]\n",
|
|
entry_flush, l1d_misses_total, entry_flush ? '>' : '<',
|
|
entry_flush ? repetitions * l1d_misses_expected :
|
|
repetitions * l1d_misses_expected / 2,
|
|
passes, repetitions);
|
|
}
|
|
|
|
if (entry_flush == entry_flush_orig) {
|
|
entry_flush = !entry_flush_orig;
|
|
if (write_debugfs_file("powerpc/entry_flush", entry_flush) < 0) {
|
|
perror("error writing to powerpc/entry_flush debugfs file");
|
|
return 1;
|
|
}
|
|
iter = repetitions;
|
|
l1d_misses_total = 0;
|
|
passes = 0;
|
|
goto again;
|
|
}
|
|
|
|
perf_event_disable(fd);
|
|
close(fd);
|
|
|
|
set_dscr(0);
|
|
|
|
if (write_debugfs_file("powerpc/rfi_flush", rfi_flush_orig) < 0) {
|
|
perror("unable to restore original value of powerpc/rfi_flush debugfs file");
|
|
return 1;
|
|
}
|
|
|
|
if (write_debugfs_file("powerpc/entry_flush", entry_flush_orig) < 0) {
|
|
perror("unable to restore original value of powerpc/entry_flush debugfs file");
|
|
return 1;
|
|
}
|
|
|
|
return rc;
|
|
}
|
|
|
|
int main(int argc, char *argv[])
|
|
{
|
|
return test_harness(entry_flush_test, "entry_flush_test");
|
|
}
|