Commit 2a68c268 authored by Linus Torvalds's avatar Linus Torvalds

Merge tag 'linux-kselftest-next-5.13-rc1' of...

Merge tag 'linux-kselftest-next-5.13-rc1' of git://git.kernel.org/pub/scm/linux/kernel/git/shuah/linux-kselftest

Pull Kselftest updates from Shuah Khan:

 - fixes and updates to resctrl test from Fenghua Yu and Reinette Chatre

 - fixes to Kselftest documentation, framework

 - minor spelling correction in timers test

* tag 'linux-kselftest-next-5.13-rc1' of git://git.kernel.org/pub/scm/linux/kernel/git/shuah/linux-kselftest: (25 commits)
  selftests/resctrl: Change a few printed messages
  Documentation: kselftest: fix path to test module files
  selftests/resctrl: Create .gitignore to include resctrl_tests
  selftests/resctrl: Fix checking for < 0 for unsigned values
  selftests/resctrl: Fix incorrect parsing of iMC counters
  selftests/resctrl: Fix unmount resctrl FS
  selftests/resctrl: Skip the test if requested resctrl feature is not supported
  selftests/resctrl: Modularize resctrl test suite main() function
  selftests/resctrl: Don't hard code value of "no_of_bits" variable
  selftests/resctrl: Fix MBA/MBM results reporting format
  selftests/resctrl: Use resctrl/info for feature detection
  selftests/resctrl: Check for resctrl mount point only if resctrl FS is supported
  selftests/resctrl: Add config dependencies
  selftests/resctrl: Fix a printed message
  selftests/resctrl: Share show_cache_info() by CAT and CMT tests
  selftests/resctrl: Call kselftest APIs to log test results
  selftests/resctrl: Rename CQM test as CMT test
  selftests/resctrl: Fix missing options "-n" and "-p"
  selftests/resctrl: Ensure sibling CPU is not same as original CPU
  selftests/resctrl: Clean up resctrl features check
  ...
parents 55e6be65 e7507478
...@@ -239,8 +239,8 @@ using a shell script test runner. ``kselftest/module.sh`` is designed ...@@ -239,8 +239,8 @@ using a shell script test runner. ``kselftest/module.sh`` is designed
to facilitate this process. There is also a header file provided to to facilitate this process. There is also a header file provided to
assist writing kernel modules that are for use with kselftest: assist writing kernel modules that are for use with kselftest:
- ``tools/testing/kselftest/kselftest_module.h`` - ``tools/testing/selftests/kselftest_module.h``
- ``tools/testing/kselftest/kselftest/module.sh`` - ``tools/testing/selftests/kselftest/module.sh``
How to use How to use
---------- ----------
......
...@@ -74,7 +74,8 @@ ifdef building_out_of_srctree ...@@ -74,7 +74,8 @@ ifdef building_out_of_srctree
rsync -aq $(TEST_PROGS) $(TEST_PROGS_EXTENDED) $(TEST_FILES) $(OUTPUT); \ rsync -aq $(TEST_PROGS) $(TEST_PROGS_EXTENDED) $(TEST_FILES) $(OUTPUT); \
fi fi
@if [ "X$(TEST_PROGS)" != "X" ]; then \ @if [ "X$(TEST_PROGS)" != "X" ]; then \
$(call RUN_TESTS, $(TEST_GEN_PROGS) $(TEST_CUSTOM_PROGS) $(OUTPUT)/$(TEST_PROGS)) ; \ $(call RUN_TESTS, $(TEST_GEN_PROGS) $(TEST_CUSTOM_PROGS) \
$(addprefix $(OUTPUT)/,$(TEST_PROGS))) ; \
else \ else \
$(call RUN_TESTS, $(TEST_GEN_PROGS) $(TEST_CUSTOM_PROGS)); \ $(call RUN_TESTS, $(TEST_GEN_PROGS) $(TEST_CUSTOM_PROGS)); \
fi fi
......
# SPDX-License-Identifier: GPL-2.0-only
resctrl_tests
CC = $(CROSS_COMPILE)gcc CC = $(CROSS_COMPILE)gcc
CFLAGS = -g -Wall CFLAGS = -g -Wall -O2 -D_FORTIFY_SOURCE=2
SRCS=$(wildcard *.c) SRCS=$(wildcard *.c)
OBJS=$(SRCS:.c=.o) OBJS=$(SRCS:.c=.o)
......
...@@ -46,8 +46,8 @@ ARGUMENTS ...@@ -46,8 +46,8 @@ ARGUMENTS
Parameter '-h' shows usage information. Parameter '-h' shows usage information.
usage: resctrl_tests [-h] [-b "benchmark_cmd [options]"] [-t test list] [-n no_of_bits] usage: resctrl_tests [-h] [-b "benchmark_cmd [options]"] [-t test list] [-n no_of_bits]
-b benchmark_cmd [options]: run specified benchmark for MBM, MBA and CQM default benchmark is builtin fill_buf -b benchmark_cmd [options]: run specified benchmark for MBM, MBA and CMT default benchmark is builtin fill_buf
-t test list: run tests specified in the test list, e.g. -t mbm, mba, cqm, cat -t test list: run tests specified in the test list, e.g. -t mbm, mba, cmt, cat
-n no_of_bits: run cache tests using specified no of bits in cache bit mask -n no_of_bits: run cache tests using specified no of bits in cache bit mask
-p cpu_no: specify CPU number to run the test. 1 is default -p cpu_no: specify CPU number to run the test. 1 is default
-h: help -h: help
...@@ -111,7 +111,7 @@ static int get_llc_perf(unsigned long *llc_perf_miss) ...@@ -111,7 +111,7 @@ static int get_llc_perf(unsigned long *llc_perf_miss)
/* /*
* Get LLC Occupancy as reported by RESCTRL FS * Get LLC Occupancy as reported by RESCTRL FS
* For CQM, * For CMT,
* 1. If con_mon grp and mon grp given, then read from mon grp in * 1. If con_mon grp and mon grp given, then read from mon grp in
* con_mon grp * con_mon grp
* 2. If only con_mon grp given, then read from con_mon grp * 2. If only con_mon grp given, then read from con_mon grp
...@@ -182,7 +182,7 @@ int measure_cache_vals(struct resctrl_val_param *param, int bm_pid) ...@@ -182,7 +182,7 @@ int measure_cache_vals(struct resctrl_val_param *param, int bm_pid)
/* /*
* Measure cache miss from perf. * Measure cache miss from perf.
*/ */
if (!strcmp(param->resctrl_val, "cat")) { if (!strncmp(param->resctrl_val, CAT_STR, sizeof(CAT_STR))) {
ret = get_llc_perf(&llc_perf_miss); ret = get_llc_perf(&llc_perf_miss);
if (ret < 0) if (ret < 0)
return ret; return ret;
...@@ -192,7 +192,7 @@ int measure_cache_vals(struct resctrl_val_param *param, int bm_pid) ...@@ -192,7 +192,7 @@ int measure_cache_vals(struct resctrl_val_param *param, int bm_pid)
/* /*
* Measure llc occupancy from resctrl. * Measure llc occupancy from resctrl.
*/ */
if (!strcmp(param->resctrl_val, "cqm")) { if (!strncmp(param->resctrl_val, CMT_STR, sizeof(CMT_STR))) {
ret = get_llc_occu_resctrl(&llc_occu_resc); ret = get_llc_occu_resctrl(&llc_occu_resc);
if (ret < 0) if (ret < 0)
return ret; return ret;
...@@ -234,7 +234,7 @@ int cat_val(struct resctrl_val_param *param) ...@@ -234,7 +234,7 @@ int cat_val(struct resctrl_val_param *param)
if (ret) if (ret)
return ret; return ret;
if ((strcmp(resctrl_val, "cat") == 0)) { if (!strncmp(resctrl_val, CAT_STR, sizeof(CAT_STR))) {
ret = initialize_llc_perf(); ret = initialize_llc_perf();
if (ret) if (ret)
return ret; return ret;
...@@ -242,7 +242,7 @@ int cat_val(struct resctrl_val_param *param) ...@@ -242,7 +242,7 @@ int cat_val(struct resctrl_val_param *param)
/* Test runs until the callback setup() tells the test to stop. */ /* Test runs until the callback setup() tells the test to stop. */
while (1) { while (1) {
if (strcmp(resctrl_val, "cat") == 0) { if (!strncmp(resctrl_val, CAT_STR, sizeof(CAT_STR))) {
ret = param->setup(1, param); ret = param->setup(1, param);
if (ret) { if (ret) {
ret = 0; ret = 0;
...@@ -270,3 +270,45 @@ int cat_val(struct resctrl_val_param *param) ...@@ -270,3 +270,45 @@ int cat_val(struct resctrl_val_param *param)
return ret; return ret;
} }
/*
* show_cache_info: show cache test result information
* @sum_llc_val: sum of LLC cache result data
* @no_of_bits: number of bits
* @cache_span: cache span in bytes for CMT or in lines for CAT
* @max_diff: max difference
* @max_diff_percent: max difference percentage
* @num_of_runs: number of runs
* @platform: show test information on this platform
* @cmt: CMT test or CAT test
*
* Return: 0 on success. non-zero on failure.
*/
int show_cache_info(unsigned long sum_llc_val, int no_of_bits,
unsigned long cache_span, unsigned long max_diff,
unsigned long max_diff_percent, unsigned long num_of_runs,
bool platform, bool cmt)
{
unsigned long avg_llc_val = 0;
float diff_percent;
long avg_diff = 0;
int ret;
avg_llc_val = sum_llc_val / (num_of_runs - 1);
avg_diff = (long)abs(cache_span - avg_llc_val);
diff_percent = ((float)cache_span - avg_llc_val) / cache_span * 100;
ret = platform && abs((int)diff_percent) > max_diff_percent &&
(cmt ? (abs(avg_diff) > max_diff) : true);
ksft_print_msg("%s Check cache miss rate within %d%%\n",
ret ? "Fail:" : "Pass:", max_diff_percent);
ksft_print_msg("Percent diff=%d\n", abs((int)diff_percent));
ksft_print_msg("Number of bits: %d\n", no_of_bits);
ksft_print_msg("Average LLC val: %lu\n", avg_llc_val);
ksft_print_msg("Cache span (%s): %lu\n", cmt ? "bytes" : "lines",
cache_span);
return ret;
}
...@@ -17,10 +17,10 @@ ...@@ -17,10 +17,10 @@
#define MAX_DIFF_PERCENT 4 #define MAX_DIFF_PERCENT 4
#define MAX_DIFF 1000000 #define MAX_DIFF 1000000
int count_of_bits; static int count_of_bits;
char cbm_mask[256]; static char cbm_mask[256];
unsigned long long_mask; static unsigned long long_mask;
unsigned long cache_size; static unsigned long cache_size;
/* /*
* Change schemata. Write schemata to specified * Change schemata. Write schemata to specified
...@@ -52,27 +52,6 @@ static int cat_setup(int num, ...) ...@@ -52,27 +52,6 @@ static int cat_setup(int num, ...)
return ret; return ret;
} }
static void show_cache_info(unsigned long sum_llc_perf_miss, int no_of_bits,
unsigned long span)
{
unsigned long allocated_cache_lines = span / 64;
unsigned long avg_llc_perf_miss = 0;
float diff_percent;
avg_llc_perf_miss = sum_llc_perf_miss / (NUM_OF_RUNS - 1);
diff_percent = ((float)allocated_cache_lines - avg_llc_perf_miss) /
allocated_cache_lines * 100;
printf("%sok CAT: cache miss rate within %d%%\n",
!is_amd && abs((int)diff_percent) > MAX_DIFF_PERCENT ?
"not " : "", MAX_DIFF_PERCENT);
tests_run++;
printf("# Percent diff=%d\n", abs((int)diff_percent));
printf("# Number of bits: %d\n", no_of_bits);
printf("# Avg_llc_perf_miss: %lu\n", avg_llc_perf_miss);
printf("# Allocated cache lines: %lu\n", allocated_cache_lines);
}
static int check_results(struct resctrl_val_param *param) static int check_results(struct resctrl_val_param *param)
{ {
char *token_array[8], temp[512]; char *token_array[8], temp[512];
...@@ -80,7 +59,7 @@ static int check_results(struct resctrl_val_param *param) ...@@ -80,7 +59,7 @@ static int check_results(struct resctrl_val_param *param)
int runs = 0, no_of_bits = 0; int runs = 0, no_of_bits = 0;
FILE *fp; FILE *fp;
printf("# Checking for pass/fail\n"); ksft_print_msg("Checking for pass/fail\n");
fp = fopen(param->filename, "r"); fp = fopen(param->filename, "r");
if (!fp) { if (!fp) {
perror("# Cannot open file"); perror("# Cannot open file");
...@@ -108,9 +87,9 @@ static int check_results(struct resctrl_val_param *param) ...@@ -108,9 +87,9 @@ static int check_results(struct resctrl_val_param *param)
fclose(fp); fclose(fp);
no_of_bits = count_bits(param->mask); no_of_bits = count_bits(param->mask);
show_cache_info(sum_llc_perf_miss, no_of_bits, param->span); return show_cache_info(sum_llc_perf_miss, no_of_bits, param->span / 64,
MAX_DIFF, MAX_DIFF_PERCENT, NUM_OF_RUNS,
return 0; !is_amd, false);
} }
void cat_test_cleanup(void) void cat_test_cleanup(void)
...@@ -132,11 +111,8 @@ int cat_perf_miss_val(int cpu_no, int n, char *cache_type) ...@@ -132,11 +111,8 @@ int cat_perf_miss_val(int cpu_no, int n, char *cache_type)
if (ret) if (ret)
return ret; return ret;
if (!validate_resctrl_feature_request("cat"))
return -1;
/* Get default cbm mask for L3/L2 cache */ /* Get default cbm mask for L3/L2 cache */
ret = get_cbm_mask(cache_type); ret = get_cbm_mask(cache_type, cbm_mask);
if (ret) if (ret)
return ret; return ret;
...@@ -146,14 +122,17 @@ int cat_perf_miss_val(int cpu_no, int n, char *cache_type) ...@@ -146,14 +122,17 @@ int cat_perf_miss_val(int cpu_no, int n, char *cache_type)
ret = get_cache_size(cpu_no, cache_type, &cache_size); ret = get_cache_size(cpu_no, cache_type, &cache_size);
if (ret) if (ret)
return ret; return ret;
printf("cache size :%lu\n", cache_size); ksft_print_msg("Cache size :%lu\n", cache_size);
/* Get max number of bits from default-cabm mask */ /* Get max number of bits from default-cabm mask */
count_of_bits = count_bits(long_mask); count_of_bits = count_bits(long_mask);
if (n < 1 || n > count_of_bits - 1) { if (!n)
printf("Invalid input value for no_of_bits n!\n"); n = count_of_bits / 2;
printf("Please Enter value in range 1 to %d\n",
if (n > count_of_bits - 1) {
ksft_print_msg("Invalid input value for no_of_bits n!\n");
ksft_print_msg("Please enter value in range 1 to %d\n",
count_of_bits - 1); count_of_bits - 1);
return -1; return -1;
} }
...@@ -164,7 +143,7 @@ int cat_perf_miss_val(int cpu_no, int n, char *cache_type) ...@@ -164,7 +143,7 @@ int cat_perf_miss_val(int cpu_no, int n, char *cache_type)
return -1; return -1;
struct resctrl_val_param param = { struct resctrl_val_param param = {
.resctrl_val = "cat", .resctrl_val = CAT_STR,
.cpu_no = cpu_no, .cpu_no = cpu_no,
.mum_resctrlfs = 0, .mum_resctrlfs = 0,
.setup = cat_setup, .setup = cat_setup,
......
// SPDX-License-Identifier: GPL-2.0 // SPDX-License-Identifier: GPL-2.0
/* /*
* Cache Monitoring Technology (CQM) test * Cache Monitoring Technology (CMT) test
* *
* Copyright (C) 2018 Intel Corporation * Copyright (C) 2018 Intel Corporation
* *
...@@ -11,17 +11,17 @@ ...@@ -11,17 +11,17 @@
#include "resctrl.h" #include "resctrl.h"
#include <unistd.h> #include <unistd.h>
#define RESULT_FILE_NAME "result_cqm" #define RESULT_FILE_NAME "result_cmt"
#define NUM_OF_RUNS 5 #define NUM_OF_RUNS 5
#define MAX_DIFF 2000000 #define MAX_DIFF 2000000
#define MAX_DIFF_PERCENT 15 #define MAX_DIFF_PERCENT 15
int count_of_bits; static int count_of_bits;
char cbm_mask[256]; static char cbm_mask[256];
unsigned long long_mask; static unsigned long long_mask;
unsigned long cache_size; static unsigned long cache_size;
static int cqm_setup(int num, ...) static int cmt_setup(int num, ...)
{ {
struct resctrl_val_param *p; struct resctrl_val_param *p;
va_list param; va_list param;
...@@ -39,38 +39,6 @@ static int cqm_setup(int num, ...) ...@@ -39,38 +39,6 @@ static int cqm_setup(int num, ...)
return 0; return 0;
} }
static void show_cache_info(unsigned long sum_llc_occu_resc, int no_of_bits,
unsigned long span)
{
unsigned long avg_llc_occu_resc = 0;
float diff_percent;
long avg_diff = 0;
bool res;
avg_llc_occu_resc = sum_llc_occu_resc / (NUM_OF_RUNS - 1);
avg_diff = (long)abs(span - avg_llc_occu_resc);
diff_percent = (((float)span - avg_llc_occu_resc) / span) * 100;
if ((abs((int)diff_percent) <= MAX_DIFF_PERCENT) ||
(abs(avg_diff) <= MAX_DIFF))
res = true;
else
res = false;
printf("%sok CQM: diff within %d, %d\%%\n", res ? "" : "not",
MAX_DIFF, (int)MAX_DIFF_PERCENT);
printf("# diff: %ld\n", avg_diff);
printf("# percent diff=%d\n", abs((int)diff_percent));
printf("# Results are displayed in (Bytes)\n");
printf("# Number of bits: %d\n", no_of_bits);
printf("# Avg_llc_occu_resc: %lu\n", avg_llc_occu_resc);
printf("# llc_occu_exp (span): %lu\n", span);
tests_run++;
}
static int check_results(struct resctrl_val_param *param, int no_of_bits) static int check_results(struct resctrl_val_param *param, int no_of_bits)
{ {
char *token_array[8], temp[512]; char *token_array[8], temp[512];
...@@ -78,7 +46,7 @@ static int check_results(struct resctrl_val_param *param, int no_of_bits) ...@@ -78,7 +46,7 @@ static int check_results(struct resctrl_val_param *param, int no_of_bits)
int runs = 0; int runs = 0;
FILE *fp; FILE *fp;
printf("# checking for pass/fail\n"); ksft_print_msg("Checking for pass/fail\n");
fp = fopen(param->filename, "r"); fp = fopen(param->filename, "r");
if (!fp) { if (!fp) {
perror("# Error in opening file\n"); perror("# Error in opening file\n");
...@@ -86,7 +54,7 @@ static int check_results(struct resctrl_val_param *param, int no_of_bits) ...@@ -86,7 +54,7 @@ static int check_results(struct resctrl_val_param *param, int no_of_bits)
return errno; return errno;
} }
while (fgets(temp, 1024, fp)) { while (fgets(temp, sizeof(temp), fp)) {
char *token = strtok(temp, ":\t"); char *token = strtok(temp, ":\t");
int fields = 0; int fields = 0;
...@@ -101,17 +69,18 @@ static int check_results(struct resctrl_val_param *param, int no_of_bits) ...@@ -101,17 +69,18 @@ static int check_results(struct resctrl_val_param *param, int no_of_bits)
runs++; runs++;
} }
fclose(fp); fclose(fp);
show_cache_info(sum_llc_occu_resc, no_of_bits, param->span);
return 0; return show_cache_info(sum_llc_occu_resc, no_of_bits, param->span,
MAX_DIFF, MAX_DIFF_PERCENT, NUM_OF_RUNS,
true, true);
} }
void cqm_test_cleanup(void) void cmt_test_cleanup(void)
{ {
remove(RESULT_FILE_NAME); remove(RESULT_FILE_NAME);
} }
int cqm_resctrl_val(int cpu_no, int n, char **benchmark_cmd) int cmt_resctrl_val(int cpu_no, int n, char **benchmark_cmd)
{ {
int ret, mum_resctrlfs; int ret, mum_resctrlfs;
...@@ -122,10 +91,10 @@ int cqm_resctrl_val(int cpu_no, int n, char **benchmark_cmd) ...@@ -122,10 +91,10 @@ int cqm_resctrl_val(int cpu_no, int n, char **benchmark_cmd)
if (ret) if (ret)
return ret; return ret;
if (!validate_resctrl_feature_request("cqm")) if (!validate_resctrl_feature_request(CMT_STR))
return -1; return -1;
ret = get_cbm_mask("L3"); ret = get_cbm_mask("L3", cbm_mask);
if (ret) if (ret)
return ret; return ret;
...@@ -134,18 +103,18 @@ int cqm_resctrl_val(int cpu_no, int n, char **benchmark_cmd) ...@@ -134,18 +103,18 @@ int cqm_resctrl_val(int cpu_no, int n, char **benchmark_cmd)
ret = get_cache_size(cpu_no, "L3", &cache_size); ret = get_cache_size(cpu_no, "L3", &cache_size);
if (ret) if (ret)
return ret; return ret;
printf("cache size :%lu\n", cache_size); ksft_print_msg("Cache size :%lu\n", cache_size);
count_of_bits = count_bits(long_mask); count_of_bits = count_bits(long_mask);
if (n < 1 || n > count_of_bits) { if (n < 1 || n > count_of_bits) {
printf("Invalid input value for numbr_of_bits n!\n"); ksft_print_msg("Invalid input value for numbr_of_bits n!\n");
printf("Please Enter value in range 1 to %d\n", count_of_bits); ksft_print_msg("Please enter value in range 1 to %d\n", count_of_bits);
return -1; return -1;
} }
struct resctrl_val_param param = { struct resctrl_val_param param = {
.resctrl_val = "cqm", .resctrl_val = CMT_STR,
.ctrlgrp = "c1", .ctrlgrp = "c1",
.mongrp = "m1", .mongrp = "m1",
.cpu_no = cpu_no, .cpu_no = cpu_no,
...@@ -154,7 +123,7 @@ int cqm_resctrl_val(int cpu_no, int n, char **benchmark_cmd) ...@@ -154,7 +123,7 @@ int cqm_resctrl_val(int cpu_no, int n, char **benchmark_cmd)
.mask = ~(long_mask << n) & long_mask, .mask = ~(long_mask << n) & long_mask,
.span = cache_size * n / count_of_bits, .span = cache_size * n / count_of_bits,
.num_of_runs = 0, .num_of_runs = 0,
.setup = cqm_setup, .setup = cmt_setup,
}; };
if (strcmp(benchmark_cmd[0], "fill_buf") == 0) if (strcmp(benchmark_cmd[0], "fill_buf") == 0)
...@@ -170,7 +139,7 @@ int cqm_resctrl_val(int cpu_no, int n, char **benchmark_cmd) ...@@ -170,7 +139,7 @@ int cqm_resctrl_val(int cpu_no, int n, char **benchmark_cmd)
if (ret) if (ret)
return ret; return ret;
cqm_test_cleanup(); cmt_test_cleanup();
return 0; return 0;
} }
CONFIG_X86_CPU_RESCTRL=y
CONFIG_PROC_CPU_RESCTRL=y
...@@ -115,7 +115,7 @@ static int fill_cache_read(unsigned char *start_ptr, unsigned char *end_ptr, ...@@ -115,7 +115,7 @@ static int fill_cache_read(unsigned char *start_ptr, unsigned char *end_ptr,
while (1) { while (1) {
ret = fill_one_span_read(start_ptr, end_ptr); ret = fill_one_span_read(start_ptr, end_ptr);
if (!strcmp(resctrl_val, "cat")) if (!strncmp(resctrl_val, CAT_STR, sizeof(CAT_STR)))
break; break;
} }
...@@ -134,7 +134,7 @@ static int fill_cache_write(unsigned char *start_ptr, unsigned char *end_ptr, ...@@ -134,7 +134,7 @@ static int fill_cache_write(unsigned char *start_ptr, unsigned char *end_ptr,
{ {
while (1) { while (1) {
fill_one_span_write(start_ptr, end_ptr); fill_one_span_write(start_ptr, end_ptr);
if (!strcmp(resctrl_val, "cat")) if (!strncmp(resctrl_val, CAT_STR, sizeof(CAT_STR)))
break; break;
} }
......
...@@ -12,7 +12,7 @@ ...@@ -12,7 +12,7 @@
#define RESULT_FILE_NAME "result_mba" #define RESULT_FILE_NAME "result_mba"
#define NUM_OF_RUNS 5 #define NUM_OF_RUNS 5
#define MAX_DIFF 300 #define MAX_DIFF_PERCENT 5
#define ALLOCATION_MAX 100 #define ALLOCATION_MAX 100
#define ALLOCATION_MIN 10 #define ALLOCATION_MIN 10
#define ALLOCATION_STEP 10 #define ALLOCATION_STEP 10
...@@ -56,13 +56,14 @@ static void show_mba_info(unsigned long *bw_imc, unsigned long *bw_resc) ...@@ -56,13 +56,14 @@ static void show_mba_info(unsigned long *bw_imc, unsigned long *bw_resc)
int allocation, runs; int allocation, runs;
bool failed = false; bool failed = false;
printf("# Results are displayed in (MB)\n"); ksft_print_msg("Results are displayed in (MB)\n");
/* Memory bandwidth from 100% down to 10% */ /* Memory bandwidth from 100% down to 10% */
for (allocation = 0; allocation < ALLOCATION_MAX / ALLOCATION_STEP; for (allocation = 0; allocation < ALLOCATION_MAX / ALLOCATION_STEP;
allocation++) { allocation++) {
unsigned long avg_bw_imc, avg_bw_resc; unsigned long avg_bw_imc, avg_bw_resc;
unsigned long sum_bw_imc = 0, sum_bw_resc = 0; unsigned long sum_bw_imc = 0, sum_bw_resc = 0;
unsigned long avg_diff; int avg_diff_per;
float avg_diff;
/* /*
* The first run is discarded due to inaccurate value from * The first run is discarded due to inaccurate value from
...@@ -76,23 +77,26 @@ static void show_mba_info(unsigned long *bw_imc, unsigned long *bw_resc) ...@@ -76,23 +77,26 @@ static void show_mba_info(unsigned long *bw_imc, unsigned long *bw_resc)
avg_bw_imc = sum_bw_imc / (NUM_OF_RUNS - 1); avg_bw_imc = sum_bw_imc / (NUM_OF_RUNS - 1);
avg_bw_resc = sum_bw_resc / (NUM_OF_RUNS - 1); avg_bw_resc = sum_bw_resc / (NUM_OF_RUNS - 1);
avg_diff = labs((long)(avg_bw_resc - avg_bw_imc)); avg_diff = (float)labs(avg_bw_resc - avg_bw_imc) / avg_bw_imc;
avg_diff_per = (int)(avg_diff * 100);
printf("%sok MBA schemata percentage %u smaller than %d %%\n",
avg_diff > MAX_DIFF ? "not " : "", ksft_print_msg("%s Check MBA diff within %d%% for schemata %u\n",
ALLOCATION_MAX - ALLOCATION_STEP * allocation, avg_diff_per > MAX_DIFF_PERCENT ?
MAX_DIFF); "Fail:" : "Pass:",
tests_run++; MAX_DIFF_PERCENT,
printf("# avg_diff: %lu\n", avg_diff); ALLOCATION_MAX - ALLOCATION_STEP * allocation);
printf("# avg_bw_imc: %lu\n", avg_bw_imc);
printf("# avg_bw_resc: %lu\n", avg_bw_resc); ksft_print_msg("avg_diff_per: %d%%\n", avg_diff_per);
if (avg_diff > MAX_DIFF) ksft_print_msg("avg_bw_imc: %lu\n", avg_bw_imc);
ksft_print_msg("avg_bw_resc: %lu\n", avg_bw_resc);
if (avg_diff_per > MAX_DIFF_PERCENT)
failed = true; failed = true;
} }
printf("%sok schemata change using MBA%s\n", failed ? "not " : "", ksft_print_msg("%s Check schemata change using MBA\n",
failed ? " # at least one test failed" : ""); failed ? "Fail:" : "Pass:");
tests_run++; if (failed)
ksft_print_msg("At least one test failed\n");
} }
static int check_results(void) static int check_results(void)
...@@ -141,7 +145,7 @@ void mba_test_cleanup(void) ...@@ -141,7 +145,7 @@ void mba_test_cleanup(void)
int mba_schemata_change(int cpu_no, char *bw_report, char **benchmark_cmd) int mba_schemata_change(int cpu_no, char *bw_report, char **benchmark_cmd)
{ {
struct resctrl_val_param param = { struct resctrl_val_param param = {
.resctrl_val = "mba", .resctrl_val = MBA_STR,
.ctrlgrp = "c1", .ctrlgrp = "c1",
.mongrp = "m1", .mongrp = "m1",
.cpu_no = cpu_no, .cpu_no = cpu_no,
...@@ -154,9 +158,6 @@ int mba_schemata_change(int cpu_no, char *bw_report, char **benchmark_cmd) ...@@ -154,9 +158,6 @@ int mba_schemata_change(int cpu_no, char *bw_report, char **benchmark_cmd)
remove(RESULT_FILE_NAME); remove(RESULT_FILE_NAME);
if (!validate_resctrl_feature_request("mba"))
return -1;
ret = resctrl_val(benchmark_cmd, &param); ret = resctrl_val(benchmark_cmd, &param);
if (ret) if (ret)
return ret; return ret;
......
...@@ -11,16 +11,16 @@ ...@@ -11,16 +11,16 @@
#include "resctrl.h" #include "resctrl.h"
#define RESULT_FILE_NAME "result_mbm" #define RESULT_FILE_NAME "result_mbm"
#define MAX_DIFF 300 #define MAX_DIFF_PERCENT 5
#define NUM_OF_RUNS 5 #define NUM_OF_RUNS 5
static void static int
show_bw_info(unsigned long *bw_imc, unsigned long *bw_resc, int span) show_bw_info(unsigned long *bw_imc, unsigned long *bw_resc, int span)
{ {
unsigned long avg_bw_imc = 0, avg_bw_resc = 0; unsigned long avg_bw_imc = 0, avg_bw_resc = 0;
unsigned long sum_bw_imc = 0, sum_bw_resc = 0; unsigned long sum_bw_imc = 0, sum_bw_resc = 0;
long avg_diff = 0; int runs, ret, avg_diff_per;
int runs; float avg_diff = 0;
/* /*
* Discard the first value which is inaccurate due to monitoring setup * Discard the first value which is inaccurate due to monitoring setup
...@@ -33,15 +33,18 @@ show_bw_info(unsigned long *bw_imc, unsigned long *bw_resc, int span) ...@@ -33,15 +33,18 @@ show_bw_info(unsigned long *bw_imc, unsigned long *bw_resc, int span)
avg_bw_imc = sum_bw_imc / 4; avg_bw_imc = sum_bw_imc / 4;
avg_bw_resc = sum_bw_resc / 4; avg_bw_resc = sum_bw_resc / 4;
avg_diff = avg_bw_resc - avg_bw_imc; avg_diff = (float)labs(avg_bw_resc - avg_bw_imc) / avg_bw_imc;
avg_diff_per = (int)(avg_diff * 100);
printf("%sok MBM: diff within %d%%\n",
labs(avg_diff) > MAX_DIFF ? "not " : "", MAX_DIFF); ret = avg_diff_per > MAX_DIFF_PERCENT;
tests_run++; ksft_print_msg("%s Check MBM diff within %d%%\n",
printf("# avg_diff: %lu\n", labs(avg_diff)); ret ? "Fail:" : "Pass:", MAX_DIFF_PERCENT);
printf("# Span (MB): %d\n", span); ksft_print_msg("avg_diff_per: %d%%\n", avg_diff_per);
printf("# avg_bw_imc: %lu\n", avg_bw_imc); ksft_print_msg("Span (MB): %d\n", span);
printf("# avg_bw_resc: %lu\n", avg_bw_resc); ksft_print_msg("avg_bw_imc: %lu\n", avg_bw_imc);
ksft_print_msg("avg_bw_resc: %lu\n", avg_bw_resc);
return ret;
} }
static int check_results(int span) static int check_results(int span)
...@@ -49,10 +52,10 @@ static int check_results(int span) ...@@ -49,10 +52,10 @@ static int check_results(int span)
unsigned long bw_imc[NUM_OF_RUNS], bw_resc[NUM_OF_RUNS]; unsigned long bw_imc[NUM_OF_RUNS], bw_resc[NUM_OF_RUNS];
char temp[1024], *token_array[8]; char temp[1024], *token_array[8];
char output[] = RESULT_FILE_NAME; char output[] = RESULT_FILE_NAME;
int runs; int runs, ret;
FILE *fp; FILE *fp;
printf("# Checking for pass/fail\n"); ksft_print_msg("Checking for pass/fail\n");
fp = fopen(output, "r"); fp = fopen(output, "r");
if (!fp) { if (!fp) {
...@@ -76,11 +79,11 @@ static int check_results(int span) ...@@ -76,11 +79,11 @@ static int check_results(int span)
runs++; runs++;
} }
show_bw_info(bw_imc, bw_resc, span); ret = show_bw_info(bw_imc, bw_resc, span);
fclose(fp); fclose(fp);
return 0; return ret;
} }
static int mbm_setup(int num, ...) static int mbm_setup(int num, ...)
...@@ -114,7 +117,7 @@ void mbm_test_cleanup(void) ...@@ -114,7 +117,7 @@ void mbm_test_cleanup(void)
int mbm_bw_change(int span, int cpu_no, char *bw_report, char **benchmark_cmd) int mbm_bw_change(int span, int cpu_no, char *bw_report, char **benchmark_cmd)
{ {
struct resctrl_val_param param = { struct resctrl_val_param param = {
.resctrl_val = "mbm", .resctrl_val = MBM_STR,
.ctrlgrp = "c1", .ctrlgrp = "c1",
.mongrp = "m1", .mongrp = "m1",
.span = span, .span = span,
...@@ -128,9 +131,6 @@ int mbm_bw_change(int span, int cpu_no, char *bw_report, char **benchmark_cmd) ...@@ -128,9 +131,6 @@ int mbm_bw_change(int span, int cpu_no, char *bw_report, char **benchmark_cmd)
remove(RESULT_FILE_NAME); remove(RESULT_FILE_NAME);
if (!validate_resctrl_feature_request("mbm"))
return -1;
ret = resctrl_val(benchmark_cmd, &param); ret = resctrl_val(benchmark_cmd, &param);
if (ret) if (ret)
return ret; return ret;
......
...@@ -23,11 +23,16 @@ ...@@ -23,11 +23,16 @@
#include <sys/eventfd.h> #include <sys/eventfd.h>
#include <asm/unistd.h> #include <asm/unistd.h>
#include <linux/perf_event.h> #include <linux/perf_event.h>
#include "../kselftest.h"
#define MB (1024 * 1024) #define MB (1024 * 1024)
#define RESCTRL_PATH "/sys/fs/resctrl" #define RESCTRL_PATH "/sys/fs/resctrl"
#define PHYS_ID_PATH "/sys/devices/system/cpu/cpu" #define PHYS_ID_PATH "/sys/devices/system/cpu/cpu"
#define CBM_MASK_PATH "/sys/fs/resctrl/info" #define CBM_MASK_PATH "/sys/fs/resctrl/info"
#define L3_PATH "/sys/fs/resctrl/info/L3"
#define MB_PATH "/sys/fs/resctrl/info/MB"
#define L3_MON_PATH "/sys/fs/resctrl/info/L3_MON"
#define L3_MON_FEATURES_PATH "/sys/fs/resctrl/info/L3_MON/mon_features"
#define PARENT_EXIT(err_msg) \ #define PARENT_EXIT(err_msg) \
do { \ do { \
...@@ -62,11 +67,15 @@ struct resctrl_val_param { ...@@ -62,11 +67,15 @@ struct resctrl_val_param {
int (*setup)(int num, ...); int (*setup)(int num, ...);
}; };
pid_t bm_pid, ppid; #define MBM_STR "mbm"
int tests_run; #define MBA_STR "mba"
#define CMT_STR "cmt"
#define CAT_STR "cat"
char llc_occup_path[1024]; extern pid_t bm_pid, ppid;
bool is_amd;
extern char llc_occup_path[1024];
extern bool is_amd;
bool check_resctrlfs_support(void); bool check_resctrlfs_support(void);
int filter_dmesg(void); int filter_dmesg(void);
...@@ -74,7 +83,7 @@ int remount_resctrlfs(bool mum_resctrlfs); ...@@ -74,7 +83,7 @@ int remount_resctrlfs(bool mum_resctrlfs);
int get_resource_id(int cpu_no, int *resource_id); int get_resource_id(int cpu_no, int *resource_id);
int umount_resctrlfs(void); int umount_resctrlfs(void);
int validate_bw_report_request(char *bw_report); int validate_bw_report_request(char *bw_report);
bool validate_resctrl_feature_request(char *resctrl_val); bool validate_resctrl_feature_request(const char *resctrl_val);
char *fgrep(FILE *inf, const char *str); char *fgrep(FILE *inf, const char *str);
int taskset_benchmark(pid_t bm_pid, int cpu_no); int taskset_benchmark(pid_t bm_pid, int cpu_no);
void run_benchmark(int signum, siginfo_t *info, void *ucontext); void run_benchmark(int signum, siginfo_t *info, void *ucontext);
...@@ -92,16 +101,20 @@ void tests_cleanup(void); ...@@ -92,16 +101,20 @@ void tests_cleanup(void);
void mbm_test_cleanup(void); void mbm_test_cleanup(void);
int mba_schemata_change(int cpu_no, char *bw_report, char **benchmark_cmd); int mba_schemata_change(int cpu_no, char *bw_report, char **benchmark_cmd);
void mba_test_cleanup(void); void mba_test_cleanup(void);
int get_cbm_mask(char *cache_type); int get_cbm_mask(char *cache_type, char *cbm_mask);
int get_cache_size(int cpu_no, char *cache_type, unsigned long *cache_size); int get_cache_size(int cpu_no, char *cache_type, unsigned long *cache_size);
void ctrlc_handler(int signum, siginfo_t *info, void *ptr); void ctrlc_handler(int signum, siginfo_t *info, void *ptr);
int cat_val(struct resctrl_val_param *param); int cat_val(struct resctrl_val_param *param);
void cat_test_cleanup(void); void cat_test_cleanup(void);
int cat_perf_miss_val(int cpu_no, int no_of_bits, char *cache_type); int cat_perf_miss_val(int cpu_no, int no_of_bits, char *cache_type);
int cqm_resctrl_val(int cpu_no, int n, char **benchmark_cmd); int cmt_resctrl_val(int cpu_no, int n, char **benchmark_cmd);
unsigned int count_bits(unsigned long n); unsigned int count_bits(unsigned long n);
void cqm_test_cleanup(void); void cmt_test_cleanup(void);
int get_core_sibling(int cpu_no); int get_core_sibling(int cpu_no);
int measure_cache_vals(struct resctrl_val_param *param, int bm_pid); int measure_cache_vals(struct resctrl_val_param *param, int bm_pid);
int show_cache_info(unsigned long sum_llc_val, int no_of_bits,
unsigned long cache_span, unsigned long max_diff,
unsigned long max_diff_percent, unsigned long num_of_runs,
bool platform, bool cmt);
#endif /* RESCTRL_H */ #endif /* RESCTRL_H */
...@@ -37,10 +37,10 @@ void detect_amd(void) ...@@ -37,10 +37,10 @@ void detect_amd(void)
static void cmd_help(void) static void cmd_help(void)
{ {
printf("usage: resctrl_tests [-h] [-b \"benchmark_cmd [options]\"] [-t test list] [-n no_of_bits]\n"); printf("usage: resctrl_tests [-h] [-b \"benchmark_cmd [options]\"] [-t test list] [-n no_of_bits]\n");
printf("\t-b benchmark_cmd [options]: run specified benchmark for MBM, MBA and CQM"); printf("\t-b benchmark_cmd [options]: run specified benchmark for MBM, MBA and CMT\n");
printf("\t default benchmark is builtin fill_buf\n"); printf("\t default benchmark is builtin fill_buf\n");
printf("\t-t test list: run tests specified in the test list, "); printf("\t-t test list: run tests specified in the test list, ");
printf("e.g. -t mbm, mba, cqm, cat\n"); printf("e.g. -t mbm, mba, cmt, cat\n");
printf("\t-n no_of_bits: run cache tests using specified no of bits in cache bit mask\n"); printf("\t-n no_of_bits: run cache tests using specified no of bits in cache bit mask\n");
printf("\t-p cpu_no: specify CPU number to run the test. 1 is default\n"); printf("\t-p cpu_no: specify CPU number to run the test. 1 is default\n");
printf("\t-h: help\n"); printf("\t-h: help\n");
...@@ -50,17 +50,88 @@ void tests_cleanup(void) ...@@ -50,17 +50,88 @@ void tests_cleanup(void)
{ {
mbm_test_cleanup(); mbm_test_cleanup();
mba_test_cleanup(); mba_test_cleanup();
cqm_test_cleanup(); cmt_test_cleanup();
cat_test_cleanup();
}
static void run_mbm_test(bool has_ben, char **benchmark_cmd, int span,
int cpu_no, char *bw_report)
{
int res;
ksft_print_msg("Starting MBM BW change ...\n");
if (!validate_resctrl_feature_request(MBM_STR)) {
ksft_test_result_skip("Hardware does not support MBM or MBM is disabled\n");
return;
}
if (!has_ben)
sprintf(benchmark_cmd[5], "%s", MBA_STR);
res = mbm_bw_change(span, cpu_no, bw_report, benchmark_cmd);
ksft_test_result(!res, "MBM: bw change\n");
mbm_test_cleanup();
}
static void run_mba_test(bool has_ben, char **benchmark_cmd, int span,
int cpu_no, char *bw_report)
{
int res;
ksft_print_msg("Starting MBA Schemata change ...\n");
if (!validate_resctrl_feature_request(MBA_STR)) {
ksft_test_result_skip("Hardware does not support MBA or MBA is disabled\n");
return;
}
if (!has_ben)
sprintf(benchmark_cmd[1], "%d", span);
res = mba_schemata_change(cpu_no, bw_report, benchmark_cmd);
ksft_test_result(!res, "MBA: schemata change\n");
mba_test_cleanup();
}
static void run_cmt_test(bool has_ben, char **benchmark_cmd, int cpu_no)
{
int res;
ksft_print_msg("Starting CMT test ...\n");
if (!validate_resctrl_feature_request(CMT_STR)) {
ksft_test_result_skip("Hardware does not support CMT or CMT is disabled\n");
return;
}
if (!has_ben)
sprintf(benchmark_cmd[5], "%s", CMT_STR);
res = cmt_resctrl_val(cpu_no, 5, benchmark_cmd);
ksft_test_result(!res, "CMT: test\n");
cmt_test_cleanup();
}
static void run_cat_test(int cpu_no, int no_of_bits)
{
int res;
ksft_print_msg("Starting CAT test ...\n");
if (!validate_resctrl_feature_request(CAT_STR)) {
ksft_test_result_skip("Hardware does not support CAT or CAT is disabled\n");
return;
}
res = cat_perf_miss_val(cpu_no, no_of_bits, "L3");
ksft_test_result(!res, "CAT: test\n");
cat_test_cleanup(); cat_test_cleanup();
} }
int main(int argc, char **argv) int main(int argc, char **argv)
{ {
bool has_ben = false, mbm_test = true, mba_test = true, cqm_test = true; bool has_ben = false, mbm_test = true, mba_test = true, cmt_test = true;
int res, c, cpu_no = 1, span = 250, argc_new = argc, i, no_of_bits = 5; int c, cpu_no = 1, span = 250, argc_new = argc, i, no_of_bits = 0;
char *benchmark_cmd[BENCHMARK_ARGS], bw_report[64], bm_type[64]; char *benchmark_cmd[BENCHMARK_ARGS], bw_report[64], bm_type[64];
char benchmark_cmd_area[BENCHMARK_ARGS][BENCHMARK_ARG_SIZE]; char benchmark_cmd_area[BENCHMARK_ARGS][BENCHMARK_ARG_SIZE];
int ben_ind, ben_count; int ben_ind, ben_count, tests = 0;
bool cat_test = true; bool cat_test = true;
for (i = 0; i < argc; i++) { for (i = 0; i < argc; i++) {
...@@ -73,7 +144,7 @@ int main(int argc, char **argv) ...@@ -73,7 +144,7 @@ int main(int argc, char **argv)
} }
} }
while ((c = getopt(argc_new, argv, "ht:b:")) != -1) { while ((c = getopt(argc_new, argv, "ht:b:n:p:")) != -1) {
char *token; char *token;
switch (c) { switch (c) {
...@@ -82,17 +153,21 @@ int main(int argc, char **argv) ...@@ -82,17 +153,21 @@ int main(int argc, char **argv)
mbm_test = false; mbm_test = false;
mba_test = false; mba_test = false;
cqm_test = false; cmt_test = false;
cat_test = false; cat_test = false;
while (token) { while (token) {
if (!strcmp(token, "mbm")) { if (!strncmp(token, MBM_STR, sizeof(MBM_STR))) {
mbm_test = true; mbm_test = true;
} else if (!strcmp(token, "mba")) { tests++;
} else if (!strncmp(token, MBA_STR, sizeof(MBA_STR))) {
mba_test = true; mba_test = true;
} else if (!strcmp(token, "cqm")) { tests++;
cqm_test = true; } else if (!strncmp(token, CMT_STR, sizeof(CMT_STR))) {
} else if (!strcmp(token, "cat")) { cmt_test = true;
tests++;
} else if (!strncmp(token, CAT_STR, sizeof(CAT_STR))) {
cat_test = true; cat_test = true;
tests++;
} else { } else {
printf("invalid argument\n"); printf("invalid argument\n");
...@@ -106,6 +181,10 @@ int main(int argc, char **argv) ...@@ -106,6 +181,10 @@ int main(int argc, char **argv)
break; break;
case 'n': case 'n':
no_of_bits = atoi(optarg); no_of_bits = atoi(optarg);
if (no_of_bits <= 0) {
printf("Bail out! invalid argument for no_of_bits\n");
return -1;
}
break; break;
case 'h': case 'h':
cmd_help(); cmd_help();
...@@ -118,7 +197,7 @@ int main(int argc, char **argv) ...@@ -118,7 +197,7 @@ int main(int argc, char **argv)
} }
} }
printf("TAP version 13\n"); ksft_print_header();
/* /*
* Typically we need root privileges, because: * Typically we need root privileges, because:
...@@ -126,7 +205,7 @@ int main(int argc, char **argv) ...@@ -126,7 +205,7 @@ int main(int argc, char **argv)
* 2. We execute perf commands * 2. We execute perf commands
*/ */
if (geteuid() != 0) if (geteuid() != 0)
printf("# WARNING: not running as root, tests may fail.\n"); return ksft_exit_fail_msg("Not running as root, abort testing.\n");
/* Detect AMD vendor */ /* Detect AMD vendor */
detect_amd(); detect_amd();
...@@ -155,48 +234,26 @@ int main(int argc, char **argv) ...@@ -155,48 +234,26 @@ int main(int argc, char **argv)
sprintf(bw_report, "reads"); sprintf(bw_report, "reads");
sprintf(bm_type, "fill_buf"); sprintf(bm_type, "fill_buf");
check_resctrlfs_support(); if (!check_resctrlfs_support())
return ksft_exit_fail_msg("resctrl FS does not exist\n");
filter_dmesg(); filter_dmesg();
if (!is_amd && mbm_test) { ksft_set_plan(tests ? : 4);
printf("# Starting MBM BW change ...\n");
if (!has_ben)
sprintf(benchmark_cmd[5], "%s", "mba");
res = mbm_bw_change(span, cpu_no, bw_report, benchmark_cmd);
printf("%sok MBM: bw change\n", res ? "not " : "");
mbm_test_cleanup();
tests_run++;
}
if (!is_amd && mba_test) { if (!is_amd && mbm_test)
printf("# Starting MBA Schemata change ...\n"); run_mbm_test(has_ben, benchmark_cmd, span, cpu_no, bw_report);
if (!has_ben)
sprintf(benchmark_cmd[1], "%d", span);
res = mba_schemata_change(cpu_no, bw_report, benchmark_cmd);
printf("%sok MBA: schemata change\n", res ? "not " : "");
mba_test_cleanup();
tests_run++;
}
if (cqm_test) { if (!is_amd && mba_test)
printf("# Starting CQM test ...\n"); run_mba_test(has_ben, benchmark_cmd, span, cpu_no, bw_report);
if (!has_ben)
sprintf(benchmark_cmd[5], "%s", "cqm");
res = cqm_resctrl_val(cpu_no, no_of_bits, benchmark_cmd);
printf("%sok CQM: test\n", res ? "not " : "");
cqm_test_cleanup();
tests_run++;
}
if (cat_test) { if (cmt_test)
printf("# Starting CAT test ...\n"); run_cmt_test(has_ben, benchmark_cmd, cpu_no);
res = cat_perf_miss_val(cpu_no, no_of_bits, "L3");
printf("%sok CAT: test\n", res ? "not " : "");
tests_run++;
cat_test_cleanup();
}
printf("1..%d\n", tests_run); if (cat_test)
run_cat_test(cpu_no, no_of_bits);
return 0; umount_resctrlfs();
return ksft_exit_pass();
} }
...@@ -221,8 +221,8 @@ static int read_from_imc_dir(char *imc_dir, int count) ...@@ -221,8 +221,8 @@ static int read_from_imc_dir(char *imc_dir, int count)
*/ */
static int num_of_imcs(void) static int num_of_imcs(void)
{ {
char imc_dir[512], *temp;
unsigned int count = 0; unsigned int count = 0;
char imc_dir[512];
struct dirent *ep; struct dirent *ep;
int ret; int ret;
DIR *dp; DIR *dp;
...@@ -230,7 +230,25 @@ static int num_of_imcs(void) ...@@ -230,7 +230,25 @@ static int num_of_imcs(void)
dp = opendir(DYN_PMU_PATH); dp = opendir(DYN_PMU_PATH);
if (dp) { if (dp) {
while ((ep = readdir(dp))) { while ((ep = readdir(dp))) {
if (strstr(ep->d_name, UNCORE_IMC)) { temp = strstr(ep->d_name, UNCORE_IMC);
if (!temp)
continue;
/*
* imc counters are named as "uncore_imc_<n>", hence
* increment the pointer to point to <n>. Note that
* sizeof(UNCORE_IMC) would count for null character as
* well and hence the last underscore character in
* uncore_imc'_' need not be counted.
*/
temp = temp + sizeof(UNCORE_IMC);
/*
* Some directories under "DYN_PMU_PATH" could have
* names like "uncore_imc_free_running", hence, check if
* first character is a numerical digit or not.
*/
if (temp[0] >= '0' && temp[0] <= '9') {
sprintf(imc_dir, "%s/%s/", DYN_PMU_PATH, sprintf(imc_dir, "%s/%s/", DYN_PMU_PATH,
ep->d_name); ep->d_name);
ret = read_from_imc_dir(imc_dir, count); ret = read_from_imc_dir(imc_dir, count);
...@@ -282,9 +300,9 @@ static int initialize_mem_bw_imc(void) ...@@ -282,9 +300,9 @@ static int initialize_mem_bw_imc(void)
* Memory B/W utilized by a process on a socket can be calculated using * Memory B/W utilized by a process on a socket can be calculated using
* iMC counters. Perf events are used to read these counters. * iMC counters. Perf events are used to read these counters.
* *
* Return: >= 0 on success. < 0 on failure. * Return: = 0 on success. < 0 on failure.
*/ */
static float get_mem_bw_imc(int cpu_no, char *bw_report) static int get_mem_bw_imc(int cpu_no, char *bw_report, float *bw_imc)
{ {
float reads, writes, of_mul_read, of_mul_write; float reads, writes, of_mul_read, of_mul_write;
int imc, j, ret; int imc, j, ret;
...@@ -355,13 +373,18 @@ static float get_mem_bw_imc(int cpu_no, char *bw_report) ...@@ -355,13 +373,18 @@ static float get_mem_bw_imc(int cpu_no, char *bw_report)
close(imc_counters_config[imc][WRITE].fd); close(imc_counters_config[imc][WRITE].fd);
} }
if (strcmp(bw_report, "reads") == 0) if (strcmp(bw_report, "reads") == 0) {
return reads; *bw_imc = reads;
return 0;
}
if (strcmp(bw_report, "writes") == 0) if (strcmp(bw_report, "writes") == 0) {
return writes; *bw_imc = writes;
return 0;
}
return (reads + writes); *bw_imc = reads + writes;
return 0;
} }
void set_mbm_path(const char *ctrlgrp, const char *mongrp, int resource_id) void set_mbm_path(const char *ctrlgrp, const char *mongrp, int resource_id)
...@@ -397,10 +420,10 @@ static void initialize_mem_bw_resctrl(const char *ctrlgrp, const char *mongrp, ...@@ -397,10 +420,10 @@ static void initialize_mem_bw_resctrl(const char *ctrlgrp, const char *mongrp,
return; return;
} }
if (strcmp(resctrl_val, "mbm") == 0) if (!strncmp(resctrl_val, MBM_STR, sizeof(MBM_STR)))
set_mbm_path(ctrlgrp, mongrp, resource_id); set_mbm_path(ctrlgrp, mongrp, resource_id);
if ((strcmp(resctrl_val, "mba") == 0)) { if (!strncmp(resctrl_val, MBA_STR, sizeof(MBA_STR))) {
if (ctrlgrp) if (ctrlgrp)
sprintf(mbm_total_path, CON_MBM_LOCAL_BYTES_PATH, sprintf(mbm_total_path, CON_MBM_LOCAL_BYTES_PATH,
RESCTRL_PATH, ctrlgrp, resource_id); RESCTRL_PATH, ctrlgrp, resource_id);
...@@ -420,9 +443,8 @@ static void initialize_mem_bw_resctrl(const char *ctrlgrp, const char *mongrp, ...@@ -420,9 +443,8 @@ static void initialize_mem_bw_resctrl(const char *ctrlgrp, const char *mongrp,
* 1. If con_mon grp is given, then read from it * 1. If con_mon grp is given, then read from it
* 2. If con_mon grp is not given, then read from root con_mon grp * 2. If con_mon grp is not given, then read from root con_mon grp
*/ */
static unsigned long get_mem_bw_resctrl(void) static int get_mem_bw_resctrl(unsigned long *mbm_total)
{ {
unsigned long mbm_total = 0;
FILE *fp; FILE *fp;
fp = fopen(mbm_total_path, "r"); fp = fopen(mbm_total_path, "r");
...@@ -431,7 +453,7 @@ static unsigned long get_mem_bw_resctrl(void) ...@@ -431,7 +453,7 @@ static unsigned long get_mem_bw_resctrl(void)
return -1; return -1;
} }
if (fscanf(fp, "%lu", &mbm_total) <= 0) { if (fscanf(fp, "%lu", mbm_total) <= 0) {
perror("Could not get mbm local bytes"); perror("Could not get mbm local bytes");
fclose(fp); fclose(fp);
...@@ -439,7 +461,7 @@ static unsigned long get_mem_bw_resctrl(void) ...@@ -439,7 +461,7 @@ static unsigned long get_mem_bw_resctrl(void)
} }
fclose(fp); fclose(fp);
return mbm_total; return 0;
} }
pid_t bm_pid, ppid; pid_t bm_pid, ppid;
...@@ -449,7 +471,7 @@ void ctrlc_handler(int signum, siginfo_t *info, void *ptr) ...@@ -449,7 +471,7 @@ void ctrlc_handler(int signum, siginfo_t *info, void *ptr)
kill(bm_pid, SIGKILL); kill(bm_pid, SIGKILL);
umount_resctrlfs(); umount_resctrlfs();
tests_cleanup(); tests_cleanup();
printf("Ending\n\n"); ksft_print_msg("Ending\n\n");
exit(EXIT_SUCCESS); exit(EXIT_SUCCESS);
} }
...@@ -492,7 +514,7 @@ static int print_results_bw(char *filename, int bm_pid, float bw_imc, ...@@ -492,7 +514,7 @@ static int print_results_bw(char *filename, int bm_pid, float bw_imc,
return 0; return 0;
} }
static void set_cqm_path(const char *ctrlgrp, const char *mongrp, char sock_num) static void set_cmt_path(const char *ctrlgrp, const char *mongrp, char sock_num)
{ {
if (strlen(ctrlgrp) && strlen(mongrp)) if (strlen(ctrlgrp) && strlen(mongrp))
sprintf(llc_occup_path, CON_MON_LCC_OCCUP_PATH, RESCTRL_PATH, sprintf(llc_occup_path, CON_MON_LCC_OCCUP_PATH, RESCTRL_PATH,
...@@ -512,7 +534,7 @@ static void set_cqm_path(const char *ctrlgrp, const char *mongrp, char sock_num) ...@@ -512,7 +534,7 @@ static void set_cqm_path(const char *ctrlgrp, const char *mongrp, char sock_num)
* @ctrlgrp: Name of the control monitor group (con_mon grp) * @ctrlgrp: Name of the control monitor group (con_mon grp)
* @mongrp: Name of the monitor group (mon grp) * @mongrp: Name of the monitor group (mon grp)
* @cpu_no: CPU number that the benchmark PID is binded to * @cpu_no: CPU number that the benchmark PID is binded to
* @resctrl_val: Resctrl feature (Eg: cat, cqm.. etc) * @resctrl_val: Resctrl feature (Eg: cat, cmt.. etc)
*/ */
static void initialize_llc_occu_resctrl(const char *ctrlgrp, const char *mongrp, static void initialize_llc_occu_resctrl(const char *ctrlgrp, const char *mongrp,
int cpu_no, char *resctrl_val) int cpu_no, char *resctrl_val)
...@@ -524,14 +546,15 @@ static void initialize_llc_occu_resctrl(const char *ctrlgrp, const char *mongrp, ...@@ -524,14 +546,15 @@ static void initialize_llc_occu_resctrl(const char *ctrlgrp, const char *mongrp,
return; return;
} }
if (strcmp(resctrl_val, "cqm") == 0) if (!strncmp(resctrl_val, CMT_STR, sizeof(CMT_STR)))
set_cqm_path(ctrlgrp, mongrp, resource_id); set_cmt_path(ctrlgrp, mongrp, resource_id);
} }
static int static int
measure_vals(struct resctrl_val_param *param, unsigned long *bw_resc_start) measure_vals(struct resctrl_val_param *param, unsigned long *bw_resc_start)
{ {
unsigned long bw_imc, bw_resc, bw_resc_end; unsigned long bw_resc, bw_resc_end;
float bw_imc;
int ret; int ret;
/* /*
...@@ -541,13 +564,13 @@ measure_vals(struct resctrl_val_param *param, unsigned long *bw_resc_start) ...@@ -541,13 +564,13 @@ measure_vals(struct resctrl_val_param *param, unsigned long *bw_resc_start)
* Compare the two values to validate resctrl value. * Compare the two values to validate resctrl value.
* It takes 1sec to measure the data. * It takes 1sec to measure the data.
*/ */
bw_imc = get_mem_bw_imc(param->cpu_no, param->bw_report); ret = get_mem_bw_imc(param->cpu_no, param->bw_report, &bw_imc);
if (bw_imc <= 0) if (ret < 0)
return bw_imc; return ret;
bw_resc_end = get_mem_bw_resctrl(); ret = get_mem_bw_resctrl(&bw_resc_end);
if (bw_resc_end <= 0) if (ret < 0)
return bw_resc_end; return ret;
bw_resc = (bw_resc_end - *bw_resc_start) / MB; bw_resc = (bw_resc_end - *bw_resc_start) / MB;
ret = print_results_bw(param->filename, bm_pid, bw_imc, bw_resc); ret = print_results_bw(param->filename, bm_pid, bw_imc, bw_resc);
...@@ -579,8 +602,8 @@ int resctrl_val(char **benchmark_cmd, struct resctrl_val_param *param) ...@@ -579,8 +602,8 @@ int resctrl_val(char **benchmark_cmd, struct resctrl_val_param *param)
if (strcmp(param->filename, "") == 0) if (strcmp(param->filename, "") == 0)
sprintf(param->filename, "stdio"); sprintf(param->filename, "stdio");
if ((strcmp(resctrl_val, "mba")) == 0 || if (!strncmp(resctrl_val, MBA_STR, sizeof(MBA_STR)) ||
(strcmp(resctrl_val, "mbm")) == 0) { !strncmp(resctrl_val, MBM_STR, sizeof(MBM_STR))) {
ret = validate_bw_report_request(param->bw_report); ret = validate_bw_report_request(param->bw_report);
if (ret) if (ret)
return ret; return ret;
...@@ -645,7 +668,7 @@ int resctrl_val(char **benchmark_cmd, struct resctrl_val_param *param) ...@@ -645,7 +668,7 @@ int resctrl_val(char **benchmark_cmd, struct resctrl_val_param *param)
PARENT_EXIT("Child is done"); PARENT_EXIT("Child is done");
} }
printf("# benchmark PID: %d\n", bm_pid); ksft_print_msg("Benchmark PID: %d\n", bm_pid);
/* /*
* Register CTRL-C handler for parent, as it has to kill benchmark * Register CTRL-C handler for parent, as it has to kill benchmark
...@@ -674,15 +697,15 @@ int resctrl_val(char **benchmark_cmd, struct resctrl_val_param *param) ...@@ -674,15 +697,15 @@ int resctrl_val(char **benchmark_cmd, struct resctrl_val_param *param)
if (ret) if (ret)
goto out; goto out;
if ((strcmp(resctrl_val, "mbm") == 0) || if (!strncmp(resctrl_val, MBM_STR, sizeof(MBM_STR)) ||
(strcmp(resctrl_val, "mba") == 0)) { !strncmp(resctrl_val, MBA_STR, sizeof(MBA_STR))) {
ret = initialize_mem_bw_imc(); ret = initialize_mem_bw_imc();
if (ret) if (ret)
goto out; goto out;
initialize_mem_bw_resctrl(param->ctrlgrp, param->mongrp, initialize_mem_bw_resctrl(param->ctrlgrp, param->mongrp,
param->cpu_no, resctrl_val); param->cpu_no, resctrl_val);
} else if (strcmp(resctrl_val, "cqm") == 0) } else if (!strncmp(resctrl_val, CMT_STR, sizeof(CMT_STR)))
initialize_llc_occu_resctrl(param->ctrlgrp, param->mongrp, initialize_llc_occu_resctrl(param->ctrlgrp, param->mongrp,
param->cpu_no, resctrl_val); param->cpu_no, resctrl_val);
...@@ -710,8 +733,8 @@ int resctrl_val(char **benchmark_cmd, struct resctrl_val_param *param) ...@@ -710,8 +733,8 @@ int resctrl_val(char **benchmark_cmd, struct resctrl_val_param *param)
/* Test runs until the callback setup() tells the test to stop. */ /* Test runs until the callback setup() tells the test to stop. */
while (1) { while (1) {
if ((strcmp(resctrl_val, "mbm") == 0) || if (!strncmp(resctrl_val, MBM_STR, sizeof(MBM_STR)) ||
(strcmp(resctrl_val, "mba") == 0)) { !strncmp(resctrl_val, MBA_STR, sizeof(MBA_STR))) {
ret = param->setup(1, param); ret = param->setup(1, param);
if (ret) { if (ret) {
ret = 0; ret = 0;
...@@ -721,7 +744,7 @@ int resctrl_val(char **benchmark_cmd, struct resctrl_val_param *param) ...@@ -721,7 +744,7 @@ int resctrl_val(char **benchmark_cmd, struct resctrl_val_param *param)
ret = measure_vals(param, &bw_resc_start); ret = measure_vals(param, &bw_resc_start);
if (ret) if (ret)
break; break;
} else if (strcmp(resctrl_val, "cqm") == 0) { } else if (!strncmp(resctrl_val, CMT_STR, sizeof(CMT_STR))) {
ret = param->setup(1, param); ret = param->setup(1, param);
if (ret) { if (ret) {
ret = 0; ret = 0;
......
...@@ -10,8 +10,6 @@ ...@@ -10,8 +10,6 @@
*/ */
#include "resctrl.h" #include "resctrl.h"
int tests_run;
static int find_resctrl_mount(char *buffer) static int find_resctrl_mount(char *buffer)
{ {
FILE *mounts; FILE *mounts;
...@@ -49,8 +47,6 @@ static int find_resctrl_mount(char *buffer) ...@@ -49,8 +47,6 @@ static int find_resctrl_mount(char *buffer)
return -ENOENT; return -ENOENT;
} }
char cbm_mask[256];
/* /*
* remount_resctrlfs - Remount resctrl FS at /sys/fs/resctrl * remount_resctrlfs - Remount resctrl FS at /sys/fs/resctrl
* @mum_resctrlfs: Should the resctrl FS be remounted? * @mum_resctrlfs: Should the resctrl FS be remounted?
...@@ -70,28 +66,25 @@ int remount_resctrlfs(bool mum_resctrlfs) ...@@ -70,28 +66,25 @@ int remount_resctrlfs(bool mum_resctrlfs)
if (ret) if (ret)
strcpy(mountpoint, RESCTRL_PATH); strcpy(mountpoint, RESCTRL_PATH);
if (!ret && mum_resctrlfs && umount(mountpoint)) { if (!ret && mum_resctrlfs && umount(mountpoint))
printf("not ok unmounting \"%s\"\n", mountpoint); ksft_print_msg("Fail: unmounting \"%s\"\n", mountpoint);
perror("# umount");
tests_run++;
}
if (!ret && !mum_resctrlfs) if (!ret && !mum_resctrlfs)
return 0; return 0;
ksft_print_msg("Mounting resctrl to \"%s\"\n", RESCTRL_PATH);
ret = mount("resctrl", RESCTRL_PATH, "resctrl", 0, NULL); ret = mount("resctrl", RESCTRL_PATH, "resctrl", 0, NULL);
printf("%sok mounting resctrl to \"%s\"\n", ret ? "not " : "",
RESCTRL_PATH);
if (ret) if (ret)
perror("# mount"); perror("# mount");
tests_run++;
return ret; return ret;
} }
int umount_resctrlfs(void) int umount_resctrlfs(void)
{ {
if (find_resctrl_mount(NULL))
return 0;
if (umount(RESCTRL_PATH)) { if (umount(RESCTRL_PATH)) {
perror("# Unable to umount resctrl"); perror("# Unable to umount resctrl");
...@@ -205,16 +198,18 @@ int get_cache_size(int cpu_no, char *cache_type, unsigned long *cache_size) ...@@ -205,16 +198,18 @@ int get_cache_size(int cpu_no, char *cache_type, unsigned long *cache_size)
/* /*
* get_cbm_mask - Get cbm mask for given cache * get_cbm_mask - Get cbm mask for given cache
* @cache_type: Cache level L2/L3 * @cache_type: Cache level L2/L3
* * @cbm_mask: cbm_mask returned as a string
* Mask is stored in cbm_mask which is global variable.
* *
* Return: = 0 on success, < 0 on failure. * Return: = 0 on success, < 0 on failure.
*/ */
int get_cbm_mask(char *cache_type) int get_cbm_mask(char *cache_type, char *cbm_mask)
{ {
char cbm_mask_path[1024]; char cbm_mask_path[1024];
FILE *fp; FILE *fp;
if (!cbm_mask)
return -1;
sprintf(cbm_mask_path, "%s/%s/cbm_mask", CBM_MASK_PATH, cache_type); sprintf(cbm_mask_path, "%s/%s/cbm_mask", CBM_MASK_PATH, cache_type);
fp = fopen(cbm_mask_path, "r"); fp = fopen(cbm_mask_path, "r");
...@@ -268,7 +263,7 @@ int get_core_sibling(int cpu_no) ...@@ -268,7 +263,7 @@ int get_core_sibling(int cpu_no)
while (token) { while (token) {
sibling_cpu_no = atoi(token); sibling_cpu_no = atoi(token);
/* Skipping core 0 as we don't want to run test on core 0 */ /* Skipping core 0 as we don't want to run test on core 0 */
if (sibling_cpu_no != 0) if (sibling_cpu_no != 0 && sibling_cpu_no != cpu_no)
break; break;
token = strtok(NULL, "-,"); token = strtok(NULL, "-,");
} }
...@@ -334,7 +329,7 @@ void run_benchmark(int signum, siginfo_t *info, void *ucontext) ...@@ -334,7 +329,7 @@ void run_benchmark(int signum, siginfo_t *info, void *ucontext)
operation = atoi(benchmark_cmd[4]); operation = atoi(benchmark_cmd[4]);
sprintf(resctrl_val, "%s", benchmark_cmd[5]); sprintf(resctrl_val, "%s", benchmark_cmd[5]);
if (strcmp(resctrl_val, "cqm") != 0) if (strncmp(resctrl_val, CMT_STR, sizeof(CMT_STR)))
buffer_span = span * MB; buffer_span = span * MB;
else else
buffer_span = span; buffer_span = span;
...@@ -458,9 +453,9 @@ int write_bm_pid_to_resctrl(pid_t bm_pid, char *ctrlgrp, char *mongrp, ...@@ -458,9 +453,9 @@ int write_bm_pid_to_resctrl(pid_t bm_pid, char *ctrlgrp, char *mongrp,
if (ret) if (ret)
goto out; goto out;
/* Create mon grp and write pid into it for "mbm" and "cqm" test */ /* Create mon grp and write pid into it for "mbm" and "cmt" test */
if ((strcmp(resctrl_val, "cqm") == 0) || if (!strncmp(resctrl_val, CMT_STR, sizeof(CMT_STR)) ||
(strcmp(resctrl_val, "mbm") == 0)) { !strncmp(resctrl_val, MBM_STR, sizeof(MBM_STR))) {
if (strlen(mongrp)) { if (strlen(mongrp)) {
sprintf(monitorgroup_p, "%s/mon_groups", controlgroup); sprintf(monitorgroup_p, "%s/mon_groups", controlgroup);
sprintf(monitorgroup, "%s/%s", monitorgroup_p, mongrp); sprintf(monitorgroup, "%s/%s", monitorgroup_p, mongrp);
...@@ -477,13 +472,10 @@ int write_bm_pid_to_resctrl(pid_t bm_pid, char *ctrlgrp, char *mongrp, ...@@ -477,13 +472,10 @@ int write_bm_pid_to_resctrl(pid_t bm_pid, char *ctrlgrp, char *mongrp,
} }
out: out:
printf("%sok writing benchmark parameters to resctrl FS\n", ksft_print_msg("Writing benchmark parameters to resctrl FS\n");
ret ? "not " : "");
if (ret) if (ret)
perror("# writing to resctrlfs"); perror("# writing to resctrlfs");
tests_run++;
return ret; return ret;
} }
...@@ -505,13 +497,13 @@ int write_schemata(char *ctrlgrp, char *schemata, int cpu_no, char *resctrl_val) ...@@ -505,13 +497,13 @@ int write_schemata(char *ctrlgrp, char *schemata, int cpu_no, char *resctrl_val)
int resource_id, ret = 0; int resource_id, ret = 0;
FILE *fp; FILE *fp;
if ((strcmp(resctrl_val, "mba") != 0) && if (strncmp(resctrl_val, MBA_STR, sizeof(MBA_STR)) &&
(strcmp(resctrl_val, "cat") != 0) && strncmp(resctrl_val, CAT_STR, sizeof(CAT_STR)) &&
(strcmp(resctrl_val, "cqm") != 0)) strncmp(resctrl_val, CMT_STR, sizeof(CMT_STR)))
return -ENOENT; return -ENOENT;
if (!schemata) { if (!schemata) {
printf("# Skipping empty schemata update\n"); ksft_print_msg("Skipping empty schemata update\n");
return -1; return -1;
} }
...@@ -528,9 +520,10 @@ int write_schemata(char *ctrlgrp, char *schemata, int cpu_no, char *resctrl_val) ...@@ -528,9 +520,10 @@ int write_schemata(char *ctrlgrp, char *schemata, int cpu_no, char *resctrl_val)
else else
sprintf(controlgroup, "%s/schemata", RESCTRL_PATH); sprintf(controlgroup, "%s/schemata", RESCTRL_PATH);
if (!strcmp(resctrl_val, "cat") || !strcmp(resctrl_val, "cqm")) if (!strncmp(resctrl_val, CAT_STR, sizeof(CAT_STR)) ||
!strncmp(resctrl_val, CMT_STR, sizeof(CMT_STR)))
sprintf(schema, "%s%d%c%s", "L3:", resource_id, '=', schemata); sprintf(schema, "%s%d%c%s", "L3:", resource_id, '=', schemata);
if (strcmp(resctrl_val, "mba") == 0) if (!strncmp(resctrl_val, MBA_STR, sizeof(MBA_STR)))
sprintf(schema, "%s%d%c%s", "MB:", resource_id, '=', schemata); sprintf(schema, "%s%d%c%s", "MB:", resource_id, '=', schemata);
fp = fopen(controlgroup, "w"); fp = fopen(controlgroup, "w");
...@@ -551,10 +544,9 @@ int write_schemata(char *ctrlgrp, char *schemata, int cpu_no, char *resctrl_val) ...@@ -551,10 +544,9 @@ int write_schemata(char *ctrlgrp, char *schemata, int cpu_no, char *resctrl_val)
fclose(fp); fclose(fp);
out: out:
printf("%sok Write schema \"%s\" to resctrl FS%s%s\n", ksft_print_msg("Write schema \"%s\" to resctrl FS%s%s\n",
ret ? "not " : "", schema, ret ? " # " : "", schema, ret ? " # " : "",
ret ? reason : ""); ret ? reason : "");
tests_run++;
return ret; return ret;
} }
...@@ -578,17 +570,19 @@ bool check_resctrlfs_support(void) ...@@ -578,17 +570,19 @@ bool check_resctrlfs_support(void)
fclose(inf); fclose(inf);
printf("%sok kernel supports resctrl filesystem\n", ret ? "" : "not "); ksft_print_msg("%s Check kernel supports resctrl filesystem\n",
tests_run++; ret ? "Pass:" : "Fail:");
if (!ret)
return ret;
dp = opendir(RESCTRL_PATH); dp = opendir(RESCTRL_PATH);
printf("%sok resctrl mountpoint \"%s\" exists\n", ksft_print_msg("%s Check resctrl mountpoint \"%s\" exists\n",
dp ? "" : "not ", RESCTRL_PATH); dp ? "Pass:" : "Fail:", RESCTRL_PATH);
if (dp) if (dp)
closedir(dp); closedir(dp);
tests_run++;
printf("# resctrl filesystem %s mounted\n", ksft_print_msg("resctrl filesystem %s mounted\n",
find_resctrl_mount(NULL) ? "not" : "is"); find_resctrl_mount(NULL) ? "not" : "is");
return ret; return ret;
...@@ -615,26 +609,56 @@ char *fgrep(FILE *inf, const char *str) ...@@ -615,26 +609,56 @@ char *fgrep(FILE *inf, const char *str)
* validate_resctrl_feature_request - Check if requested feature is valid. * validate_resctrl_feature_request - Check if requested feature is valid.
* @resctrl_val: Requested feature * @resctrl_val: Requested feature
* *
* Return: 0 on success, non-zero on failure * Return: True if the feature is supported, else false
*/ */
bool validate_resctrl_feature_request(char *resctrl_val) bool validate_resctrl_feature_request(const char *resctrl_val)
{ {
FILE *inf = fopen("/proc/cpuinfo", "r"); struct stat statbuf;
bool found = false; bool found = false;
char *res; char *res;
FILE *inf;
if (!inf) if (!resctrl_val)
return false; return false;
res = fgrep(inf, "flags"); if (remount_resctrlfs(false))
return false;
if (!strncmp(resctrl_val, CAT_STR, sizeof(CAT_STR))) {
if (!stat(L3_PATH, &statbuf))
return true;
} else if (!strncmp(resctrl_val, MBA_STR, sizeof(MBA_STR))) {
if (!stat(MB_PATH, &statbuf))
return true;
} else if (!strncmp(resctrl_val, MBM_STR, sizeof(MBM_STR)) ||
!strncmp(resctrl_val, CMT_STR, sizeof(CMT_STR))) {
if (!stat(L3_MON_PATH, &statbuf)) {
inf = fopen(L3_MON_FEATURES_PATH, "r");
if (!inf)
return false;
if (!strncmp(resctrl_val, CMT_STR, sizeof(CMT_STR))) {
res = fgrep(inf, "llc_occupancy");
if (res) { if (res) {
char *s = strchr(res, ':'); found = true;
free(res);
}
}
found = s && !strstr(s, resctrl_val); if (!strncmp(resctrl_val, MBM_STR, sizeof(MBM_STR))) {
res = fgrep(inf, "mbm_total_bytes");
if (res) {
free(res);
res = fgrep(inf, "mbm_local_bytes");
if (res) {
found = true;
free(res); free(res);
} }
}
}
fclose(inf); fclose(inf);
}
}
return found; return found;
} }
...@@ -671,9 +695,9 @@ int filter_dmesg(void) ...@@ -671,9 +695,9 @@ int filter_dmesg(void)
while (fgets(line, 1024, fp)) { while (fgets(line, 1024, fp)) {
if (strstr(line, "intel_rdt:")) if (strstr(line, "intel_rdt:"))
printf("# dmesg: %s", line); ksft_print_msg("dmesg: %s", line);
if (strstr(line, "resctrl:")) if (strstr(line, "resctrl:"))
printf("# dmesg: %s", line); ksft_print_msg("dmesg: %s", line);
} }
fclose(fp); fclose(fp);
waitpid(pid, NULL, 0); waitpid(pid, NULL, 0);
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment