mirror of
https://sourceware.org/git/glibc.git
synced 2025-08-05 19:35:52 +03:00
benchtests: Improve readability of JSON output
Add a small library to print JSON values and use it to improve the readability of the benchmark output and the readability of the benchmark code. ChangeLog: 2014-04-11 Will Newton <will.newton@linaro.org> * benchtests/Makefile (extra-objs): Add json-lib.o. (bench-func): Tidy up JSON output. * benchtests/bench-skeleton.c: Include json-lib.h. (main): Use JSON library functions to do output of benchmark results. * benchtests/bench-timing-type.c (main): Output the timing type simply, leaving formatting to the user. * benchtests/json-lib.c: New file. * benchtests/json-lib.h: Likewise.
This commit is contained in:
12
ChangeLog
12
ChangeLog
@@ -1,3 +1,15 @@
|
|||||||
|
2014-04-11 Will Newton <will.newton@linaro.org>
|
||||||
|
|
||||||
|
* benchtests/Makefile (extra-objs): Add json-lib.o.
|
||||||
|
(bench-func): Tidy up JSON output.
|
||||||
|
* benchtests/bench-skeleton.c: Include json-lib.h.
|
||||||
|
(main): Use JSON library functions to do output of
|
||||||
|
benchmark results.
|
||||||
|
* benchtests/bench-timing-type.c (main): Output the
|
||||||
|
timing type simply, leaving formatting to the user.
|
||||||
|
* benchtests/json-lib.c: New file.
|
||||||
|
* benchtests/json-lib.h: Likewise.
|
||||||
|
|
||||||
2014-04-11 Torvald Riegel <triegel@redhat.com>
|
2014-04-11 Torvald Riegel <triegel@redhat.com>
|
||||||
|
|
||||||
[BZ #15215]
|
[BZ #15215]
|
||||||
|
@@ -101,6 +101,8 @@ cpp-srcs-left := $(binaries-benchset:=.c) $(binaries-bench:=.c)
|
|||||||
lib := nonlib
|
lib := nonlib
|
||||||
include $(patsubst %,$(..)cppflags-iterator.mk,$(cpp-srcs-left))
|
include $(patsubst %,$(..)cppflags-iterator.mk,$(cpp-srcs-left))
|
||||||
|
|
||||||
|
extra-objs += json-lib.o
|
||||||
|
|
||||||
bench-deps := bench-skeleton.c bench-timing.h Makefile
|
bench-deps := bench-skeleton.c bench-timing.h Makefile
|
||||||
|
|
||||||
run-bench = $(test-wrapper-env) \
|
run-bench = $(test-wrapper-env) \
|
||||||
@@ -127,9 +129,9 @@ bench-set: $(binaries-benchset)
|
|||||||
# so one could even execute them individually and process it using any JSON
|
# so one could even execute them individually and process it using any JSON
|
||||||
# capable language or tool.
|
# capable language or tool.
|
||||||
bench-func: $(binaries-bench)
|
bench-func: $(binaries-bench)
|
||||||
{ echo "{"; \
|
{ timing_type=$$($(timing-type)); \
|
||||||
$(timing-type); \
|
echo "{\"timing_type\": \"$${timing_type}\","; \
|
||||||
echo " ,\"functions\": {"; \
|
echo " \"functions\": {"; \
|
||||||
for run in $^; do \
|
for run in $^; do \
|
||||||
if ! [ "x$${run}" = "x$<" ]; then \
|
if ! [ "x$${run}" = "x$<" ]; then \
|
||||||
echo ","; \
|
echo ","; \
|
||||||
@@ -137,6 +139,7 @@ bench-func: $(binaries-bench)
|
|||||||
echo "Running $${run}" >&2; \
|
echo "Running $${run}" >&2; \
|
||||||
$(run-bench) $(DETAILED_OPT); \
|
$(run-bench) $(DETAILED_OPT); \
|
||||||
done; \
|
done; \
|
||||||
|
echo; \
|
||||||
echo " }"; \
|
echo " }"; \
|
||||||
echo "}"; } > $(objpfx)bench.out-tmp; \
|
echo "}"; } > $(objpfx)bench.out-tmp; \
|
||||||
if [ -f $(objpfx)bench.out ]; then \
|
if [ -f $(objpfx)bench.out ]; then \
|
||||||
@@ -144,7 +147,7 @@ bench-func: $(binaries-bench)
|
|||||||
fi; \
|
fi; \
|
||||||
mv -f $(objpfx)bench.out-tmp $(objpfx)bench.out
|
mv -f $(objpfx)bench.out-tmp $(objpfx)bench.out
|
||||||
|
|
||||||
$(timing-type) $(binaries-bench) $(binaries-benchset): %: %.o \
|
$(timing-type) $(binaries-bench) $(binaries-benchset): %: %.o $(objpfx)json-lib.o \
|
||||||
$(sort $(filter $(common-objpfx)lib%,$(link-libc))) \
|
$(sort $(filter $(common-objpfx)lib%,$(link-libc))) \
|
||||||
$(addprefix $(csu-objpfx),start.o) $(+preinit) $(+postinit)
|
$(addprefix $(csu-objpfx),start.o) $(+preinit) $(+postinit)
|
||||||
$(+link)
|
$(+link)
|
||||||
|
@@ -23,6 +23,7 @@
|
|||||||
#include <time.h>
|
#include <time.h>
|
||||||
#include <inttypes.h>
|
#include <inttypes.h>
|
||||||
#include "bench-timing.h"
|
#include "bench-timing.h"
|
||||||
|
#include "json-lib.h"
|
||||||
|
|
||||||
volatile unsigned int dontoptimize = 0;
|
volatile unsigned int dontoptimize = 0;
|
||||||
|
|
||||||
@@ -50,6 +51,7 @@ main (int argc, char **argv)
|
|||||||
struct timespec runtime;
|
struct timespec runtime;
|
||||||
timing_t start, end;
|
timing_t start, end;
|
||||||
bool detailed = false;
|
bool detailed = false;
|
||||||
|
json_ctx_t json_ctx;
|
||||||
|
|
||||||
if (argc == 2 && !strcmp (argv[1], "-d"))
|
if (argc == 2 && !strcmp (argv[1], "-d"))
|
||||||
detailed = true;
|
detailed = true;
|
||||||
@@ -64,13 +66,13 @@ main (int argc, char **argv)
|
|||||||
|
|
||||||
iters = 1000 * res;
|
iters = 1000 * res;
|
||||||
|
|
||||||
|
json_init (&json_ctx, 2, stdout);
|
||||||
|
|
||||||
/* Begin function. */
|
/* Begin function. */
|
||||||
printf ("\"%s\": {\n", FUNCNAME);
|
json_attr_object_begin (&json_ctx, FUNCNAME);
|
||||||
|
|
||||||
for (int v = 0; v < NUM_VARIANTS; v++)
|
for (int v = 0; v < NUM_VARIANTS; v++)
|
||||||
{
|
{
|
||||||
if (v)
|
|
||||||
putc (',', stdout);
|
|
||||||
/* Run for approximately DURATION seconds. */
|
/* Run for approximately DURATION seconds. */
|
||||||
clock_gettime (CLOCK_MONOTONIC_RAW, &runtime);
|
clock_gettime (CLOCK_MONOTONIC_RAW, &runtime);
|
||||||
runtime.tv_sec += DURATION;
|
runtime.tv_sec += DURATION;
|
||||||
@@ -119,28 +121,31 @@ main (int argc, char **argv)
|
|||||||
d_total_s = total;
|
d_total_s = total;
|
||||||
d_iters = iters;
|
d_iters = iters;
|
||||||
|
|
||||||
printf ("\"%s\": {\n", VARIANT (v));
|
/* Begin variant. */
|
||||||
printf ("\"duration\": %g, \"iterations\": %g, "
|
json_attr_object_begin (&json_ctx, VARIANT (v));
|
||||||
"\"max\": %g, \"min\": %g, \"mean\": %g\n",
|
|
||||||
d_total_s, d_total_i, max / d_iters, min / d_iters,
|
json_attr_double (&json_ctx, "duration", d_total_s);
|
||||||
d_total_s / d_total_i);
|
json_attr_double (&json_ctx, "iterations", d_total_i);
|
||||||
|
json_attr_double (&json_ctx, "max", max / d_iters);
|
||||||
|
json_attr_double (&json_ctx, "min", min / d_iters);
|
||||||
|
json_attr_double (&json_ctx, "mean", d_total_s / d_total_i);
|
||||||
|
|
||||||
if (detailed)
|
if (detailed)
|
||||||
{
|
{
|
||||||
printf (",\n\"timings\": [");
|
json_array_begin (&json_ctx, "timings");
|
||||||
|
|
||||||
for (int i = 0; i < NUM_SAMPLES (v); i++)
|
for (int i = 0; i < NUM_SAMPLES (v); i++)
|
||||||
{
|
json_element_double (&json_ctx, RESULT (v, i));
|
||||||
if (i > 0)
|
|
||||||
putc (',', stdout);
|
json_array_end (&json_ctx);
|
||||||
printf ("%g", RESULT (v, i));
|
|
||||||
}
|
}
|
||||||
puts ("]");
|
|
||||||
}
|
/* End variant. */
|
||||||
puts ("}");
|
json_attr_object_end (&json_ctx);
|
||||||
}
|
}
|
||||||
|
|
||||||
/* End function. */
|
/* End function. */
|
||||||
puts ("}");
|
json_attr_object_end (&json_ctx);
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
@@ -22,6 +22,6 @@
|
|||||||
int
|
int
|
||||||
main (int argc, char **argv)
|
main (int argc, char **argv)
|
||||||
{
|
{
|
||||||
printf ("\"timing-type\": \"%s\"\n", TIMING_TYPE);
|
puts (TIMING_TYPE);
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
178
benchtests/json-lib.c
Normal file
178
benchtests/json-lib.c
Normal file
@@ -0,0 +1,178 @@
|
|||||||
|
/* Simple library for printing JSON data.
|
||||||
|
Copyright (C) 2014 Free Software Foundation, Inc.
|
||||||
|
This file is part of the GNU C Library.
|
||||||
|
|
||||||
|
The GNU C Library is free software; you can redistribute it and/or
|
||||||
|
modify it under the terms of the GNU Lesser General Public
|
||||||
|
License as published by the Free Software Foundation; either
|
||||||
|
version 2.1 of the License, or (at your option) any later version.
|
||||||
|
|
||||||
|
The GNU C Library is distributed in the hope that it will be useful,
|
||||||
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
||||||
|
Lesser General Public License for more details.
|
||||||
|
|
||||||
|
You should have received a copy of the GNU Lesser General Public
|
||||||
|
License along with the GNU C Library; if not, see
|
||||||
|
<http://www.gnu.org/licenses/>. */
|
||||||
|
|
||||||
|
#include <string.h>
|
||||||
|
|
||||||
|
#include "json-lib.h"
|
||||||
|
|
||||||
|
void
|
||||||
|
json_init (json_ctx_t *ctx, unsigned int indent_level, FILE *fp)
|
||||||
|
{
|
||||||
|
ctx->indent_level = indent_level;
|
||||||
|
ctx->fp = fp;
|
||||||
|
ctx->first_element = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
do_indent (json_ctx_t *ctx)
|
||||||
|
{
|
||||||
|
char indent_buf[ctx->indent_level + 1];
|
||||||
|
|
||||||
|
memset (indent_buf, ' ', ctx->indent_level + 1);
|
||||||
|
indent_buf[ctx->indent_level] = '\0';
|
||||||
|
|
||||||
|
fputs (indent_buf, ctx->fp);
|
||||||
|
}
|
||||||
|
|
||||||
|
void
|
||||||
|
json_document_begin (json_ctx_t *ctx)
|
||||||
|
{
|
||||||
|
do_indent (ctx);
|
||||||
|
|
||||||
|
fputs ("{\n", ctx->fp);
|
||||||
|
|
||||||
|
ctx->indent_level++;
|
||||||
|
ctx->first_element = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
void
|
||||||
|
json_document_end (json_ctx_t *ctx)
|
||||||
|
{
|
||||||
|
ctx->indent_level--;
|
||||||
|
|
||||||
|
do_indent (ctx);
|
||||||
|
|
||||||
|
fputs ("\n}", ctx->fp);
|
||||||
|
}
|
||||||
|
|
||||||
|
void
|
||||||
|
json_attr_object_begin (json_ctx_t *ctx, const char *name)
|
||||||
|
{
|
||||||
|
if (!ctx->first_element)
|
||||||
|
fprintf (ctx->fp, ",\n");
|
||||||
|
|
||||||
|
do_indent (ctx);
|
||||||
|
|
||||||
|
fprintf (ctx->fp, "\"%s\": {\n", name);
|
||||||
|
|
||||||
|
ctx->indent_level++;
|
||||||
|
ctx->first_element = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
void
|
||||||
|
json_attr_object_end (json_ctx_t *ctx)
|
||||||
|
{
|
||||||
|
ctx->indent_level--;
|
||||||
|
ctx->first_element = false;
|
||||||
|
|
||||||
|
fputs ("\n", ctx->fp);
|
||||||
|
|
||||||
|
do_indent (ctx);
|
||||||
|
|
||||||
|
fputs ("}", ctx->fp);
|
||||||
|
}
|
||||||
|
|
||||||
|
void
|
||||||
|
json_attr_string (json_ctx_t *ctx, const char *name, const char *s)
|
||||||
|
{
|
||||||
|
if (!ctx->first_element)
|
||||||
|
fprintf (ctx->fp, ",\n");
|
||||||
|
else
|
||||||
|
ctx->first_element = false;
|
||||||
|
|
||||||
|
do_indent (ctx);
|
||||||
|
|
||||||
|
fprintf (ctx->fp, "\"%s\": \"%s\"", name, s);
|
||||||
|
}
|
||||||
|
|
||||||
|
void
|
||||||
|
json_attr_double (json_ctx_t *ctx, const char *name, double d)
|
||||||
|
{
|
||||||
|
if (!ctx->first_element)
|
||||||
|
fprintf (ctx->fp, ",\n");
|
||||||
|
else
|
||||||
|
ctx->first_element = false;
|
||||||
|
|
||||||
|
do_indent (ctx);
|
||||||
|
|
||||||
|
fprintf (ctx->fp, "\"%s\": %g", name, d);
|
||||||
|
}
|
||||||
|
|
||||||
|
void
|
||||||
|
json_array_begin (json_ctx_t *ctx, const char *name)
|
||||||
|
{
|
||||||
|
if (!ctx->first_element)
|
||||||
|
fprintf (ctx->fp, ",\n");
|
||||||
|
|
||||||
|
do_indent (ctx);
|
||||||
|
|
||||||
|
fprintf (ctx->fp, "\"%s\": [", name);
|
||||||
|
|
||||||
|
ctx->indent_level++;
|
||||||
|
ctx->first_element = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
void
|
||||||
|
json_array_end (json_ctx_t *ctx)
|
||||||
|
{
|
||||||
|
ctx->indent_level--;
|
||||||
|
ctx->first_element = false;
|
||||||
|
|
||||||
|
fputs ("]", ctx->fp);
|
||||||
|
}
|
||||||
|
|
||||||
|
void
|
||||||
|
json_element_double (json_ctx_t *ctx, double d)
|
||||||
|
{
|
||||||
|
if (!ctx->first_element)
|
||||||
|
fprintf (ctx->fp, ", %g", d);
|
||||||
|
else
|
||||||
|
{
|
||||||
|
fprintf (ctx->fp, "%g", d);
|
||||||
|
ctx->first_element = false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void
|
||||||
|
json_element_object_begin (json_ctx_t *ctx)
|
||||||
|
{
|
||||||
|
if (!ctx->first_element)
|
||||||
|
fprintf (ctx->fp, ",");
|
||||||
|
|
||||||
|
fputs ("\n", ctx->fp);
|
||||||
|
|
||||||
|
do_indent (ctx);
|
||||||
|
|
||||||
|
fputs ("{\n", ctx->fp);
|
||||||
|
|
||||||
|
ctx->indent_level++;
|
||||||
|
ctx->first_element = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
void
|
||||||
|
json_element_object_end (json_ctx_t *ctx)
|
||||||
|
{
|
||||||
|
ctx->indent_level--;
|
||||||
|
ctx->first_element = false;
|
||||||
|
|
||||||
|
fputs ("\n", ctx->fp);
|
||||||
|
|
||||||
|
do_indent (ctx);
|
||||||
|
|
||||||
|
fputs ("}", ctx->fp);
|
||||||
|
}
|
47
benchtests/json-lib.h
Normal file
47
benchtests/json-lib.h
Normal file
@@ -0,0 +1,47 @@
|
|||||||
|
/* Simple library for printing JSON data.
|
||||||
|
Copyright (C) 2014 Free Software Foundation, Inc.
|
||||||
|
This file is part of the GNU C Library.
|
||||||
|
|
||||||
|
The GNU C Library is free software; you can redistribute it and/or
|
||||||
|
modify it under the terms of the GNU Lesser General Public
|
||||||
|
License as published by the Free Software Foundation; either
|
||||||
|
version 2.1 of the License, or (at your option) any later version.
|
||||||
|
|
||||||
|
The GNU C Library is distributed in the hope that it will be useful,
|
||||||
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
||||||
|
Lesser General Public License for more details.
|
||||||
|
|
||||||
|
You should have received a copy of the GNU Lesser General Public
|
||||||
|
License along with the GNU C Library; if not, see
|
||||||
|
<http://www.gnu.org/licenses/>. */
|
||||||
|
|
||||||
|
#ifndef __JSON_LIB_H__
|
||||||
|
#define __JSON_LIB_H__
|
||||||
|
|
||||||
|
#include <stdbool.h>
|
||||||
|
#include <stdio.h>
|
||||||
|
|
||||||
|
struct json_ctx
|
||||||
|
{
|
||||||
|
FILE *fp;
|
||||||
|
unsigned int indent_level;
|
||||||
|
bool first_element;
|
||||||
|
};
|
||||||
|
|
||||||
|
typedef struct json_ctx json_ctx_t;
|
||||||
|
|
||||||
|
void json_init (json_ctx_t *ctx, unsigned int indent_level, FILE *fp);
|
||||||
|
void json_document_begin (json_ctx_t *ctx);
|
||||||
|
void json_document_end (json_ctx_t *ctx);
|
||||||
|
void json_attr_object_begin (json_ctx_t *ctx, const char *name);
|
||||||
|
void json_attr_object_end (json_ctx_t *ctx);
|
||||||
|
void json_attr_string (json_ctx_t *ctx, const char *name, const char *s);
|
||||||
|
void json_attr_double (json_ctx_t *ctx, const char *name, double d);
|
||||||
|
void json_array_begin (json_ctx_t *ctx, const char *name);
|
||||||
|
void json_array_end (json_ctx_t *ctx);
|
||||||
|
void json_element_double (json_ctx_t *ctx, double d);
|
||||||
|
void json_element_object_begin (json_ctx_t *ctx);
|
||||||
|
void json_element_object_end (json_ctx_t *ctx);
|
||||||
|
|
||||||
|
#endif
|
Reference in New Issue
Block a user