2012-10-31 14:52:47 +00:00
|
|
|
/*
|
|
|
|
* The struct perf_event_attr test support.
|
|
|
|
*
|
|
|
|
* This test is embedded inside into perf directly and is governed
|
|
|
|
* by the PERF_TEST_ATTR environment variable and hook inside
|
|
|
|
* sys_perf_event_open function.
|
|
|
|
*
|
|
|
|
* The general idea is to store 'struct perf_event_attr' details for
|
|
|
|
* each event created within single perf command. Each event details
|
|
|
|
* are stored into separate text file. Once perf command is finished
|
|
|
|
* these files can be checked for values we expect for command.
|
|
|
|
*
|
|
|
|
* Besides 'struct perf_event_attr' values we also store 'fd' and
|
|
|
|
* 'group_fd' values to allow checking for groups created.
|
|
|
|
*
|
|
|
|
* This all is triggered by setting PERF_TEST_ATTR environment variable.
|
|
|
|
* It must contain name of existing directory with access and write
|
|
|
|
* permissions. All the event text files are stored there.
|
|
|
|
*/
|
|
|
|
|
2017-06-27 14:49:13 +00:00
|
|
|
#include <debug.h>
|
2017-04-18 13:46:11 +00:00
|
|
|
#include <errno.h>
|
2017-04-17 18:23:08 +00:00
|
|
|
#include <inttypes.h>
|
2012-10-31 14:52:47 +00:00
|
|
|
#include <stdlib.h>
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <linux/types.h>
|
|
|
|
#include <linux/kernel.h>
|
2017-04-19 21:51:14 +00:00
|
|
|
#include <sys/param.h>
|
2017-04-19 23:57:47 +00:00
|
|
|
#include <sys/types.h>
|
|
|
|
#include <sys/stat.h>
|
|
|
|
#include <unistd.h>
|
2012-10-31 14:52:47 +00:00
|
|
|
#include "../perf.h"
|
2015-12-15 15:39:39 +00:00
|
|
|
#include <subcmd/exec-cmd.h>
|
2012-11-10 00:46:51 +00:00
|
|
|
#include "tests.h"
|
2012-10-31 14:52:47 +00:00
|
|
|
|
|
|
|
#define ENV "PERF_TEST_ATTR"
|
|
|
|
|
|
|
|
static char *dir;
|
2017-07-03 14:50:18 +00:00
|
|
|
static bool ready;
|
2012-10-31 14:52:47 +00:00
|
|
|
|
|
|
|
void test_attr__init(void)
|
|
|
|
{
|
|
|
|
dir = getenv(ENV);
|
|
|
|
test_attr__enabled = (dir != NULL);
|
|
|
|
}
|
|
|
|
|
|
|
|
#define BUFSIZE 1024
|
|
|
|
|
2012-11-05 15:49:37 +00:00
|
|
|
#define __WRITE_ASS(str, fmt, data) \
|
2012-10-31 14:52:47 +00:00
|
|
|
do { \
|
|
|
|
char buf[BUFSIZE]; \
|
|
|
|
size_t size; \
|
|
|
|
\
|
|
|
|
size = snprintf(buf, BUFSIZE, #str "=%"fmt "\n", data); \
|
|
|
|
if (1 != fwrite(buf, size, 1, file)) { \
|
|
|
|
perror("test attr - failed to write event file"); \
|
|
|
|
fclose(file); \
|
|
|
|
return -1; \
|
|
|
|
} \
|
|
|
|
\
|
|
|
|
} while (0)
|
|
|
|
|
2012-11-05 15:49:37 +00:00
|
|
|
#define WRITE_ASS(field, fmt) __WRITE_ASS(field, fmt, attr->field)
|
|
|
|
|
2012-10-31 14:52:47 +00:00
|
|
|
static int store_event(struct perf_event_attr *attr, pid_t pid, int cpu,
|
|
|
|
int fd, int group_fd, unsigned long flags)
|
|
|
|
{
|
|
|
|
FILE *file;
|
|
|
|
char path[PATH_MAX];
|
|
|
|
|
2017-07-03 14:50:18 +00:00
|
|
|
if (!ready)
|
|
|
|
return 0;
|
|
|
|
|
2012-10-31 14:52:47 +00:00
|
|
|
snprintf(path, PATH_MAX, "%s/event-%d-%llu-%d", dir,
|
|
|
|
attr->type, attr->config, fd);
|
|
|
|
|
|
|
|
file = fopen(path, "w+");
|
|
|
|
if (!file) {
|
|
|
|
perror("test attr - failed to open event file");
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (fprintf(file, "[event-%d-%llu-%d]\n",
|
|
|
|
attr->type, attr->config, fd) < 0) {
|
|
|
|
perror("test attr - failed to write event file");
|
|
|
|
fclose(file);
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* syscall arguments */
|
2012-11-05 15:49:37 +00:00
|
|
|
__WRITE_ASS(fd, "d", fd);
|
|
|
|
__WRITE_ASS(group_fd, "d", group_fd);
|
|
|
|
__WRITE_ASS(cpu, "d", cpu);
|
|
|
|
__WRITE_ASS(pid, "d", pid);
|
|
|
|
__WRITE_ASS(flags, "lu", flags);
|
2012-10-31 14:52:47 +00:00
|
|
|
|
|
|
|
/* struct perf_event_attr */
|
2012-11-05 15:49:37 +00:00
|
|
|
WRITE_ASS(type, PRIu32);
|
|
|
|
WRITE_ASS(size, PRIu32);
|
|
|
|
WRITE_ASS(config, "llu");
|
|
|
|
WRITE_ASS(sample_period, "llu");
|
|
|
|
WRITE_ASS(sample_type, "llu");
|
|
|
|
WRITE_ASS(read_format, "llu");
|
|
|
|
WRITE_ASS(disabled, "d");
|
|
|
|
WRITE_ASS(inherit, "d");
|
|
|
|
WRITE_ASS(pinned, "d");
|
|
|
|
WRITE_ASS(exclusive, "d");
|
|
|
|
WRITE_ASS(exclude_user, "d");
|
|
|
|
WRITE_ASS(exclude_kernel, "d");
|
|
|
|
WRITE_ASS(exclude_hv, "d");
|
|
|
|
WRITE_ASS(exclude_idle, "d");
|
|
|
|
WRITE_ASS(mmap, "d");
|
|
|
|
WRITE_ASS(comm, "d");
|
|
|
|
WRITE_ASS(freq, "d");
|
|
|
|
WRITE_ASS(inherit_stat, "d");
|
|
|
|
WRITE_ASS(enable_on_exec, "d");
|
|
|
|
WRITE_ASS(task, "d");
|
2012-11-05 15:49:38 +00:00
|
|
|
WRITE_ASS(watermark, "d");
|
2012-11-05 15:49:37 +00:00
|
|
|
WRITE_ASS(precise_ip, "d");
|
|
|
|
WRITE_ASS(mmap_data, "d");
|
|
|
|
WRITE_ASS(sample_id_all, "d");
|
|
|
|
WRITE_ASS(exclude_host, "d");
|
|
|
|
WRITE_ASS(exclude_guest, "d");
|
|
|
|
WRITE_ASS(exclude_callchain_kernel, "d");
|
|
|
|
WRITE_ASS(exclude_callchain_user, "d");
|
|
|
|
WRITE_ASS(wakeup_events, PRIu32);
|
|
|
|
WRITE_ASS(bp_type, PRIu32);
|
|
|
|
WRITE_ASS(config1, "llu");
|
|
|
|
WRITE_ASS(config2, "llu");
|
|
|
|
WRITE_ASS(branch_sample_type, "llu");
|
|
|
|
WRITE_ASS(sample_regs_user, "llu");
|
|
|
|
WRITE_ASS(sample_stack_user, PRIu32);
|
|
|
|
|
2012-10-31 14:52:47 +00:00
|
|
|
fclose(file);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
void test_attr__open(struct perf_event_attr *attr, pid_t pid, int cpu,
|
|
|
|
int fd, int group_fd, unsigned long flags)
|
|
|
|
{
|
|
|
|
int errno_saved = errno;
|
|
|
|
|
2017-07-03 14:50:17 +00:00
|
|
|
if ((fd != -1) && store_event(attr, pid, cpu, fd, group_fd, flags)) {
|
2017-06-27 14:49:13 +00:00
|
|
|
pr_err("test attr FAILED");
|
|
|
|
exit(128);
|
|
|
|
}
|
2012-10-31 14:52:47 +00:00
|
|
|
|
|
|
|
errno = errno_saved;
|
|
|
|
}
|
2012-10-30 22:02:05 +00:00
|
|
|
|
2017-07-03 14:50:18 +00:00
|
|
|
void test_attr__ready(void)
|
|
|
|
{
|
|
|
|
if (unlikely(test_attr__enabled) && !ready)
|
|
|
|
ready = true;
|
|
|
|
}
|
|
|
|
|
2012-10-30 22:02:05 +00:00
|
|
|
static int run_dir(const char *d, const char *perf)
|
|
|
|
{
|
2013-02-25 09:52:49 +00:00
|
|
|
char v[] = "-vvvvv";
|
|
|
|
int vcnt = min(verbose, (int) sizeof(v) - 1);
|
2012-10-30 22:02:05 +00:00
|
|
|
char cmd[3*PATH_MAX];
|
|
|
|
|
2017-02-17 08:17:38 +00:00
|
|
|
if (verbose > 0)
|
2013-02-25 09:52:49 +00:00
|
|
|
vcnt++;
|
|
|
|
|
|
|
|
snprintf(cmd, 3*PATH_MAX, PYTHON " %s/attr.py -d %s/attr/ -p %s %.*s",
|
|
|
|
d, d, perf, vcnt, v);
|
2012-10-30 22:02:05 +00:00
|
|
|
|
|
|
|
return system(cmd);
|
|
|
|
}
|
|
|
|
|
perf tests: Pass the subtest index to each test routine
Some tests have sub-tests we want to run, so allow passing this.
Wang tried to avoid having to touch all tests, but then, having the
test.func in an anonymous union makes the build fail on older compilers,
like the one in RHEL6, where:
test a = {
.func = foo,
};
fails.
To fix it leave the func pointer in the main structure and pass the subtest
index to all tests, end result function is the same, but we have just one
function pointer, not two, with and without the subtest index as an argument.
Cc: Adrian Hunter <adrian.hunter@intel.com>
Cc: David Ahern <dsahern@gmail.com>
Cc: Jiri Olsa <jolsa@redhat.com>
Cc: Namhyung Kim <namhyung@kernel.org>
Cc: Wang Nan <wangnan0@huawei.com>
Link: http://lkml.kernel.org/n/tip-5genj0ficwdmelpoqlds0u4y@git.kernel.org
Signed-off-by: Arnaldo Carvalho de Melo <acme@redhat.com>
2015-11-19 15:01:48 +00:00
|
|
|
int test__attr(int subtest __maybe_unused)
|
2012-10-30 22:02:05 +00:00
|
|
|
{
|
|
|
|
struct stat st;
|
|
|
|
char path_perf[PATH_MAX];
|
|
|
|
char path_dir[PATH_MAX];
|
|
|
|
|
|
|
|
/* First try developement tree tests. */
|
|
|
|
if (!lstat("./tests", &st))
|
|
|
|
return run_dir("./tests", "./perf");
|
|
|
|
|
|
|
|
/* Then installed path. */
|
2015-12-15 15:39:37 +00:00
|
|
|
snprintf(path_dir, PATH_MAX, "%s/tests", get_argv_exec_path());
|
2012-10-30 22:02:05 +00:00
|
|
|
snprintf(path_perf, PATH_MAX, "%s/perf", BINDIR);
|
|
|
|
|
|
|
|
if (!lstat(path_dir, &st) &&
|
|
|
|
!lstat(path_perf, &st))
|
|
|
|
return run_dir(path_dir, path_perf);
|
|
|
|
|
2015-11-03 10:44:42 +00:00
|
|
|
return TEST_SKIP;
|
2012-10-30 22:02:05 +00:00
|
|
|
}
|