forked from Minki/linux
selftests/powerpc: Add cache_shape sniff test
This is a very basic test of the new cache shape AUXV entries. All it does at the moment is look for the entries and error out if we don't find all the ones we expect. Primarily intended for folks bringing up a new chip to check that the cache info is making it all the way to userspace correctly. Signed-off-by: Michael Ellerman <mpe@ellerman.id.au>
This commit is contained in:
parent
e3028437cb
commit
b8b6ff0196
@ -14,6 +14,7 @@ export CFLAGS
|
||||
|
||||
SUB_DIRS = alignment \
|
||||
benchmarks \
|
||||
cache_shape \
|
||||
copyloops \
|
||||
context_switch \
|
||||
dscr \
|
||||
|
1
tools/testing/selftests/powerpc/cache_shape/.gitignore
vendored
Normal file
1
tools/testing/selftests/powerpc/cache_shape/.gitignore
vendored
Normal file
@ -0,0 +1 @@
|
||||
cache_shape
|
10
tools/testing/selftests/powerpc/cache_shape/Makefile
Normal file
10
tools/testing/selftests/powerpc/cache_shape/Makefile
Normal file
@ -0,0 +1,10 @@
|
||||
TEST_PROGS := cache_shape
|
||||
|
||||
all: $(TEST_PROGS)
|
||||
|
||||
$(TEST_PROGS): ../harness.c ../utils.c
|
||||
|
||||
include ../../lib.mk
|
||||
|
||||
clean:
|
||||
rm -f $(TEST_PROGS) *.o
|
125
tools/testing/selftests/powerpc/cache_shape/cache_shape.c
Normal file
125
tools/testing/selftests/powerpc/cache_shape/cache_shape.c
Normal file
@ -0,0 +1,125 @@
|
||||
/*
|
||||
* Copyright 2017, Michael Ellerman, IBM Corp.
|
||||
*
|
||||
* This program is free software; you can redistribute it and/or
|
||||
* modify it under the terms of the GNU General Public License
|
||||
* as published by the Free Software Foundation; either version
|
||||
* 2 of the License, or (at your option) any later version.
|
||||
*/
|
||||
|
||||
#include <elf.h>
|
||||
#include <errno.h>
|
||||
#include <fcntl.h>
|
||||
#include <link.h>
|
||||
#include <stdio.h>
|
||||
#include <stdlib.h>
|
||||
#include <string.h>
|
||||
#include <sys/stat.h>
|
||||
#include <sys/types.h>
|
||||
#include <sys/wait.h>
|
||||
#include <unistd.h>
|
||||
|
||||
#include "utils.h"
|
||||
|
||||
#ifndef AT_L1I_CACHESIZE
|
||||
#define AT_L1I_CACHESIZE 40
|
||||
#define AT_L1I_CACHEGEOMETRY 41
|
||||
#define AT_L1D_CACHESIZE 42
|
||||
#define AT_L1D_CACHEGEOMETRY 43
|
||||
#define AT_L2_CACHESIZE 44
|
||||
#define AT_L2_CACHEGEOMETRY 45
|
||||
#define AT_L3_CACHESIZE 46
|
||||
#define AT_L3_CACHEGEOMETRY 47
|
||||
#endif
|
||||
|
||||
static void print_size(const char *label, uint32_t val)
|
||||
{
|
||||
printf("%s cache size: %#10x %10dB %10dK\n", label, val, val, val / 1024);
|
||||
}
|
||||
|
||||
static void print_geo(const char *label, uint32_t val)
|
||||
{
|
||||
uint16_t assoc;
|
||||
|
||||
printf("%s line size: %#10x ", label, val & 0xFFFF);
|
||||
|
||||
assoc = val >> 16;
|
||||
if (assoc)
|
||||
printf("%u-way", assoc);
|
||||
else
|
||||
printf("fully");
|
||||
|
||||
printf(" associative\n");
|
||||
}
|
||||
|
||||
static int test_cache_shape()
|
||||
{
|
||||
static char buffer[4096];
|
||||
ElfW(auxv_t) *p;
|
||||
int found;
|
||||
|
||||
FAIL_IF(read_auxv(buffer, sizeof(buffer)));
|
||||
|
||||
found = 0;
|
||||
|
||||
p = find_auxv_entry(AT_L1I_CACHESIZE, buffer);
|
||||
if (p) {
|
||||
found++;
|
||||
print_size("L1I ", (uint32_t)p->a_un.a_val);
|
||||
}
|
||||
|
||||
p = find_auxv_entry(AT_L1I_CACHEGEOMETRY, buffer);
|
||||
if (p) {
|
||||
found++;
|
||||
print_geo("L1I ", (uint32_t)p->a_un.a_val);
|
||||
}
|
||||
|
||||
p = find_auxv_entry(AT_L1D_CACHESIZE, buffer);
|
||||
if (p) {
|
||||
found++;
|
||||
print_size("L1D ", (uint32_t)p->a_un.a_val);
|
||||
}
|
||||
|
||||
p = find_auxv_entry(AT_L1D_CACHEGEOMETRY, buffer);
|
||||
if (p) {
|
||||
found++;
|
||||
print_geo("L1D ", (uint32_t)p->a_un.a_val);
|
||||
}
|
||||
|
||||
p = find_auxv_entry(AT_L2_CACHESIZE, buffer);
|
||||
if (p) {
|
||||
found++;
|
||||
print_size("L2 ", (uint32_t)p->a_un.a_val);
|
||||
}
|
||||
|
||||
p = find_auxv_entry(AT_L2_CACHEGEOMETRY, buffer);
|
||||
if (p) {
|
||||
found++;
|
||||
print_geo("L2 ", (uint32_t)p->a_un.a_val);
|
||||
}
|
||||
|
||||
p = find_auxv_entry(AT_L3_CACHESIZE, buffer);
|
||||
if (p) {
|
||||
found++;
|
||||
print_size("L3 ", (uint32_t)p->a_un.a_val);
|
||||
}
|
||||
|
||||
p = find_auxv_entry(AT_L3_CACHEGEOMETRY, buffer);
|
||||
if (p) {
|
||||
found++;
|
||||
print_geo("L3 ", (uint32_t)p->a_un.a_val);
|
||||
}
|
||||
|
||||
/* If we found none we're probably on a system where they don't exist */
|
||||
SKIP_IF(found == 0);
|
||||
|
||||
/* But if we found any, we expect to find them all */
|
||||
FAIL_IF(found != 8);
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
int main(void)
|
||||
{
|
||||
return test_harness(test_cache_shape, "cache_shape");
|
||||
}
|
Loading…
Reference in New Issue
Block a user