When NDEBUG is defined, the assert macro will be expanded to nothing. Some assert calls used in perf are also including some functionality (e.g. system calls), not only validity checks. Therefore, if NDEBUG is defined, this functionality will be removed along with the assert. Perf also defines BUG_ON based on assert, so it has the same problem. Define BUG_ON so that the condition will be executed when NDEBUG is defined. Replace the assert statements that have these side effects with BUG_ON. For defining BUG_ON, use "if (cond) {}" insted of "if (cond) ;" because in the latter case build fails with "error: suggest braces around empty body in an ‘if’ statement [-Werror=empty-body]" Suggested-by: Peter Zijlstra <a.p.zijlstra@chello.nl> Signed-off-by: Irina Tirdea <irina.tirdea@intel.com> Reviewed-by: Namhyung Kim <namhyung@kernel.org> Reviewed-by: Pekka Enberg <penberg@kernel.org> Cc: David Ahern <dsahern@gmail.com> Cc: Ingo Molnar <mingo@redhat.com> Cc: Namhyung Kim <namhyung.kim@lge.com> Cc: Paul Mackerras <paulus@samba.org> Cc: Peter Zijlstra <a.p.zijlstra@chello.nl> Cc: Steven Rostedt <rostedt@goodmis.org> Link: http://lkml.kernel.org/r/1347082551-2394-1-git-send-email-irina.tirdea@intel.com Signed-off-by: Arnaldo Carvalho de Melo <acme@redhat.com>
128 lines
2.7 KiB
C
128 lines
2.7 KiB
C
/*
|
|
*
|
|
* sched-pipe.c
|
|
*
|
|
* pipe: Benchmark for pipe()
|
|
*
|
|
* Based on pipe-test-1m.c by Ingo Molnar <mingo@redhat.com>
|
|
* http://people.redhat.com/mingo/cfs-scheduler/tools/pipe-test-1m.c
|
|
* Ported to perf by Hitoshi Mitake <mitake@dcl.info.waseda.ac.jp>
|
|
*
|
|
*/
|
|
|
|
#include "../perf.h"
|
|
#include "../util/util.h"
|
|
#include "../util/parse-options.h"
|
|
#include "../builtin.h"
|
|
#include "bench.h"
|
|
|
|
#include <unistd.h>
|
|
#include <stdio.h>
|
|
#include <stdlib.h>
|
|
#include <signal.h>
|
|
#include <sys/wait.h>
|
|
#include <linux/unistd.h>
|
|
#include <string.h>
|
|
#include <errno.h>
|
|
#include <assert.h>
|
|
#include <sys/time.h>
|
|
#include <sys/types.h>
|
|
|
|
#define LOOPS_DEFAULT 1000000
|
|
static int loops = LOOPS_DEFAULT;
|
|
|
|
static const struct option options[] = {
|
|
OPT_INTEGER('l', "loop", &loops,
|
|
"Specify number of loops"),
|
|
OPT_END()
|
|
};
|
|
|
|
static const char * const bench_sched_pipe_usage[] = {
|
|
"perf bench sched pipe <options>",
|
|
NULL
|
|
};
|
|
|
|
int bench_sched_pipe(int argc, const char **argv,
|
|
const char *prefix __used)
|
|
{
|
|
int pipe_1[2], pipe_2[2];
|
|
int m = 0, i;
|
|
struct timeval start, stop, diff;
|
|
unsigned long long result_usec = 0;
|
|
|
|
/*
|
|
* why does "ret" exist?
|
|
* discarding returned value of read(), write()
|
|
* causes error in building environment for perf
|
|
*/
|
|
int __used ret, wait_stat;
|
|
pid_t pid, retpid __used;
|
|
|
|
argc = parse_options(argc, argv, options,
|
|
bench_sched_pipe_usage, 0);
|
|
|
|
BUG_ON(pipe(pipe_1));
|
|
BUG_ON(pipe(pipe_2));
|
|
|
|
pid = fork();
|
|
assert(pid >= 0);
|
|
|
|
gettimeofday(&start, NULL);
|
|
|
|
if (!pid) {
|
|
for (i = 0; i < loops; i++) {
|
|
ret = read(pipe_1[0], &m, sizeof(int));
|
|
ret = write(pipe_2[1], &m, sizeof(int));
|
|
}
|
|
} else {
|
|
for (i = 0; i < loops; i++) {
|
|
ret = write(pipe_1[1], &m, sizeof(int));
|
|
ret = read(pipe_2[0], &m, sizeof(int));
|
|
}
|
|
}
|
|
|
|
gettimeofday(&stop, NULL);
|
|
timersub(&stop, &start, &diff);
|
|
|
|
if (pid) {
|
|
retpid = waitpid(pid, &wait_stat, 0);
|
|
assert((retpid == pid) && WIFEXITED(wait_stat));
|
|
} else {
|
|
exit(0);
|
|
}
|
|
|
|
switch (bench_format) {
|
|
case BENCH_FORMAT_DEFAULT:
|
|
printf("# Executed %d pipe operations between two tasks\n\n",
|
|
loops);
|
|
|
|
result_usec = diff.tv_sec * 1000000;
|
|
result_usec += diff.tv_usec;
|
|
|
|
printf(" %14s: %lu.%03lu [sec]\n\n", "Total time",
|
|
diff.tv_sec,
|
|
(unsigned long) (diff.tv_usec/1000));
|
|
|
|
printf(" %14lf usecs/op\n",
|
|
(double)result_usec / (double)loops);
|
|
printf(" %14d ops/sec\n",
|
|
(int)((double)loops /
|
|
((double)result_usec / (double)1000000)));
|
|
break;
|
|
|
|
case BENCH_FORMAT_SIMPLE:
|
|
printf("%lu.%03lu\n",
|
|
diff.tv_sec,
|
|
(unsigned long) (diff.tv_usec / 1000));
|
|
break;
|
|
|
|
default:
|
|
/* reaching here is something disaster */
|
|
fprintf(stderr, "Unknown format:%d\n", bench_format);
|
|
exit(1);
|
|
break;
|
|
}
|
|
|
|
return 0;
|
|
}
|