The following changes since commit 3449ab8c4d2addb716105ded698438a1d3811572: Default to terse version 3 (2012-09-14 23:35:08 +0200) are available in the git repository at: git://git.kernel.dk/fio.git master Jens Axboe (1): Add --output-format command line option Shaohua Li (1): Add a simple json encoder and use it to print fio output in json format Makefile | 2 +- README | 3 +- backend.c | 2 +- client.c | 2 +- diskutil.c | 62 +++++++++++- diskutil.h | 6 +- eta.c | 4 +- filesetup.c | 2 +- fio.h | 8 ++- init.c | 26 ++++- json.c | 336 +++++++++++++++++++++++++++++++++++++++++++++++++++++++++++ json.h | 77 ++++++++++++++ stat.c | 223 ++++++++++++++++++++++++++++++++++++++-- 13 files changed, 728 insertions(+), 25 deletions(-) create mode 100644 json.c create mode 100644 json.h --- Diff of recent changes: diff --git a/Makefile b/Makefile index 9d3945b..bac062c 100644 --- a/Makefile +++ b/Makefile @@ -14,7 +14,7 @@ SOURCE := gettime.c fio.c ioengines.c init.c stat.c log.c time.c filesetup.c \ rbtree.c smalloc.c filehash.c profile.c debug.c lib/rand.c \ lib/num2str.c lib/ieee754.c $(wildcard crc/*.c) engines/cpu.c \ engines/mmap.c engines/sync.c engines/null.c engines/net.c \ - memalign.c server.c client.c iolog.c backend.c libfio.c flow.c + memalign.c server.c client.c iolog.c backend.c libfio.c flow.c json.c ifeq ($(UNAME), Linux) SOURCE += diskutil.c fifo.c blktrace.c helpers.c cgroup.c trim.c \ diff --git a/README b/README index a135017..535b077 100644 --- a/README +++ b/README @@ -125,8 +125,9 @@ $ fio --latency-log Generate per-job latency logs --bandwidth-log Generate per-job bandwidth logs --minimal Minimal (terse) output - --version Print version info and exit + --output-format=type Output format (terse,json,normal) --terse-version=type Terse version output format (default 3, or 2 or 4). + --version Print version info and exit --help Print this page --cmdhelp=cmd Print command help, "all" for all of them --enghelp=engine Print ioengine help, or list available ioengines diff --git a/backend.c b/backend.c index 9b0d791..ce0a009 100644 --- a/backend.c +++ b/backend.c @@ -1381,7 +1381,7 @@ static void run_threads(void) set_sig_handlers(); - if (!terse_output) { + if (output_format == FIO_OUTPUT_NORMAL) { log_info("Starting "); if (nr_thread) log_info("%d thread%s", nr_thread, diff --git a/client.c b/client.c index 50c1704..93c7103 100644 --- a/client.c +++ b/client.c @@ -733,7 +733,7 @@ static void handle_du(struct fio_client *client, struct fio_net_cmd *cmd) log_info("\nDisk stats (read/write):\n"); } - print_disk_util(&du->dus, &du->agg, terse_output); + print_disk_util(&du->dus, &du->agg, output_format == FIO_OUTPUT_TERSE); } static void convert_jobs_eta(struct jobs_eta *je) diff --git a/diskutil.c b/diskutil.c index a3a5b4d..d2c0b97 100644 --- a/diskutil.c +++ b/diskutil.c @@ -597,10 +597,60 @@ void print_disk_util(struct disk_util_stat *dus, struct disk_util_agg *agg, log_info("\n"); } -void show_disk_util(int terse) +static void print_disk_util_json(struct disk_util *du, struct json_array *array) +{ + double util = 0; + struct disk_util_stat *dus = &du->dus; + struct disk_util_agg *agg = &du->agg; + struct json_object *obj; + + obj = json_create_object(); + json_array_add_value_object(array, obj); + + if (dus->msec) + util = (double) 100 * dus->io_ticks / (double) dus->msec; + if (util > 100.0) + util = 100.0; + + + json_object_add_value_string(obj, "name", dus->name); + json_object_add_value_int(obj, "read_ios", dus->ios[0]); + json_object_add_value_int(obj, "write_ios", dus->ios[1]); + json_object_add_value_int(obj, "read_merges", dus->merges[0]); + json_object_add_value_int(obj, "write_merges", dus->merges[1]); + json_object_add_value_int(obj, "read_ticks", dus->ticks[0]); + json_object_add_value_int(obj, "write_ticks", dus->ticks[1]); + json_object_add_value_int(obj, "in_queue", dus->time_in_queue); + json_object_add_value_float(obj, "util", util); + + /* + * If the device has slaves, aggregate the stats for + * those slave devices also. + */ + if (!agg->slavecount) + return; + json_object_add_value_int(obj, "aggr_read_ios", + agg->ios[0] / agg->slavecount); + json_object_add_value_int(obj, "aggr_write_ios", + agg->ios[1] / agg->slavecount); + json_object_add_value_int(obj, "aggr_read_merges", + agg->merges[0] / agg->slavecount); + json_object_add_value_int(obj, "aggr_write_merge", + agg->merges[1] / agg->slavecount); + json_object_add_value_int(obj, "aggr_read_ticks", + agg->ticks[0] / agg->slavecount); + json_object_add_value_int(obj, "aggr_write_ticks", + agg->ticks[1] / agg->slavecount); + json_object_add_value_int(obj, "aggr_in_queue", + agg->time_in_queue / agg->slavecount); + json_object_add_value_float(obj, "aggr_util", agg->max_util.u.f); +} + +void show_disk_util(int terse, struct json_object *parent) { struct flist_head *entry; struct disk_util *du; + struct json_array *array = NULL; fio_mutex_down(disk_util_mutex); @@ -612,11 +662,19 @@ void show_disk_util(int terse) if (!terse) log_info("\nDisk stats (read/write):\n"); + if (terse && terse_version == 4) { + array = json_create_array(); + json_object_add_value_array(parent, "disk_util", array); + } + flist_for_each(entry, &disk_list) { du = flist_entry(entry, struct disk_util, list); aggregate_slaves_stats(du); - print_disk_util(&du->dus, &du->agg, terse); + if (terse && terse_version == 4) + print_disk_util_json(du, array); + else + print_disk_util(&du->dus, &du->agg, terse); } fio_mutex_up(disk_util_mutex); diff --git a/diskutil.h b/diskutil.h index 88dde55..b223150 100644 --- a/diskutil.h +++ b/diskutil.h @@ -1,6 +1,6 @@ #ifndef FIO_DISKUTIL_H #define FIO_DISKUTIL_H - +#include "json.h" #define FIO_DU_NAME_SZ 64 /* @@ -101,14 +101,14 @@ extern void wait_for_disk_thread_exit(void); */ #ifdef FIO_HAVE_DISK_UTIL extern void print_disk_util(struct disk_util_stat *, struct disk_util_agg *, int terse); -extern void show_disk_util(int terse); +extern void show_disk_util(int terse, struct json_object *parent); extern void free_disk_util(void); extern void init_disk_util(struct thread_data *); extern int update_io_ticks(void); extern void setup_disk_util(void); #else #define print_disk_util(dus, agg, terse) -#define show_disk_util(terse) +#define show_disk_util(terse, parent) #define free_disk_util() #define init_disk_util(td) #define setup_disk_util() diff --git a/eta.c b/eta.c index 1caee49..075ce8c 100644 --- a/eta.c +++ b/eta.c @@ -264,7 +264,9 @@ int calc_thread_status(struct jobs_eta *je, int force) static struct timeval rate_prev_time, disp_prev_time; if (!force) { - if (temp_stall_ts || terse_output || eta_print == FIO_ETA_NEVER) + if (output_format != FIO_OUTPUT_NORMAL) + return 0; + if (temp_stall_ts || eta_print == FIO_ETA_NEVER) return 0; if (!isatty(STDOUT_FILENO) && (eta_print != FIO_ETA_ALWAYS)) diff --git a/filesetup.c b/filesetup.c index c284071..64da8bb 100644 --- a/filesetup.c +++ b/filesetup.c @@ -792,7 +792,7 @@ int setup_files(struct thread_data *td) */ if (need_extend) { temp_stall_ts = 1; - if (!terse_output) + if (output_format == FIO_OUTPUT_NORMAL) log_info("%s: Laying out IO file(s) (%u file(s) /" " %lluMB)\n", td->o.name, need_extend, extend_size >> 20); diff --git a/fio.h b/fio.h index 0064a1d..b2bbe93 100644 --- a/fio.h +++ b/fio.h @@ -526,7 +526,7 @@ extern unsigned int thread_number; extern unsigned int nr_process, nr_thread; extern int shm_id; extern int groupid; -extern int terse_output; +extern int output_format; extern int temp_stall_ts; extern unsigned long long mlock_size; extern uintptr_t page_mask, page_size; @@ -761,4 +761,10 @@ static inline void td_io_u_free_notify(struct thread_data *td) extern const char *fio_get_arch_string(int); extern const char *fio_get_os_string(int); +enum { + FIO_OUTPUT_TERSE = 0, + FIO_OUTPUT_JSON, + FIO_OUTPUT_NORMAL, +}; + #endif diff --git a/init.c b/init.c index 54ee1db..da1f472 100644 --- a/init.c +++ b/init.c @@ -36,7 +36,7 @@ static struct thread_data def_thread; struct thread_data *threads = NULL; int exitall_on_terminate = 0; -int terse_output = 0; +int output_format = FIO_OUTPUT_NORMAL; int eta_print; unsigned long long mlock_size = 0; FILE *f_out = NULL; @@ -117,6 +117,11 @@ static struct option l_opts[FIO_NR_OPTIONS] = { .val = 'm' | FIO_CLIENT_FLAG, }, { + .name = (char *) "output-format", + .has_arg = optional_argument, + .val = 'F' | FIO_CLIENT_FLAG, + }, + { .name = (char *) "version", .has_arg = no_argument, .val = 'v' | FIO_CLIENT_FLAG, @@ -870,7 +875,7 @@ static int add_job(struct thread_data *td, const char *jobname, int job_add_num) if (!td->o.name) td->o.name = strdup(jobname); - if (!terse_output) { + if (output_format == FIO_OUTPUT_NORMAL) { if (!job_add_num) { if (!strcmp(td->io_ops->name, "cpuio")) { log_info("%s: ioengine=cpu, cpuload=%u," @@ -1213,8 +1218,9 @@ static void usage(const char *name) printf(" --latency-log\t\tGenerate per-job latency logs\n"); printf(" --bandwidth-log\tGenerate per-job bandwidth logs\n"); printf(" --minimal\t\tMinimal (terse) output\n"); - printf(" --version\t\tPrint version info and exit\n"); + printf(" --output-format=x\tOutput format (terse,json,normal)\n"); printf(" --terse-version=x\tSet terse version output format to 'x'\n"); + printf(" --version\t\tPrint version info and exit\n"); printf(" --help\t\tPrint this page\n"); printf(" --cmdhelp=cmd\t\tPrint command help, \"all\" for all of" " them\n"); @@ -1406,7 +1412,17 @@ int parse_cmd_line(int argc, char *argv[]) f_err = f_out; break; case 'm': - terse_output = 1; + output_format = FIO_OUTPUT_TERSE; + break; + case 'F': + if (!strcmp(optarg, "minimal") || + !strcmp(optarg, "terse") || + !strcmp(optarg, "csv")) + output_format = FIO_OUTPUT_TERSE; + else if (!strcmp(optarg, "json")) + output_format = FIO_OUTPUT_JSON; + else + output_format = FIO_OUTPUT_NORMAL; break; case 'h': if (!cur_client) { @@ -1677,7 +1693,7 @@ int parse_options(int argc, char *argv[]) fio_gtod_cpu = def_thread.o.gtod_cpu; } - if (!terse_output) + if (output_format == FIO_OUTPUT_NORMAL) log_info("%s\n", fio_version_string); return 0; diff --git a/json.c b/json.c new file mode 100644 index 0000000..8efbbda --- /dev/null +++ b/json.c @@ -0,0 +1,336 @@ +#include <stdlib.h> +#include <string.h> +#include <stdio.h> +#include <errno.h> +#include <stdarg.h> +#include "json.h" +#include "log.h" + +struct json_object *json_create_object(void) +{ + struct json_object *obj = malloc(sizeof(struct json_object)); + if (obj) + memset(obj, 0, sizeof(struct json_object)); + return obj; +} + +struct json_array *json_create_array(void) +{ + struct json_array *array = malloc(sizeof(struct json_array)); + if (array) + memset(array, 0, sizeof(struct json_array)); + return array; +} + +static struct json_pair *json_create_pair(const char *name, struct json_value *value) +{ + struct json_pair *pair = malloc(sizeof(struct json_pair)); + if (pair) { + pair->name = strdup(name); + pair->value = value; + + value->parent_type = JSON_PARENT_TYPE_PAIR; + value->parent_pair = pair; + } + return pair; +} + +static struct json_value *json_create_value_int(long number) +{ + struct json_value *value = malloc(sizeof(struct json_value)); + + if (value) { + value->type = JSON_TYPE_INTEGER; + value->integer_number = number; + } + return value; +} + +static struct json_value *json_create_value_float(float number) +{ + struct json_value *value = malloc(sizeof(struct json_value)); + + if (value) { + value->type = JSON_TYPE_FLOAT; + value->float_number = number; + } + return value; +} + +static struct json_value *json_create_value_string(const char *str) +{ + struct json_value *value = malloc(sizeof(struct json_value)); + + if (value) { + value->type = JSON_TYPE_STRING; + value->string = strdup(str); + if (!value->string) { + free(value); + value = NULL; + } + } + return value; +} + +static struct json_value *json_create_value_object(struct json_object *obj) +{ + struct json_value *value = malloc(sizeof(struct json_value)); + + if (value) { + value->type = JSON_TYPE_OBJECT; + value->object = obj; + obj->parent = value; + } + return value; +} + +static struct json_value *json_create_value_array(struct json_array *array) +{ + struct json_value *value = malloc(sizeof(struct json_value)); + + if (value) { + value->type = JSON_TYPE_ARRAY; + value->array = array; + array->parent = value; + } + return value; +} + +static void json_free_pair(struct json_pair *pair); +static void json_free_value(struct json_value *value); + +void json_free_object(struct json_object *obj) +{ + int i; + + for (i = 0; i < obj->pair_cnt; i++) + json_free_pair(obj->pairs[i]); + free(obj->pairs); + free(obj); +} + +static void json_free_array(struct json_array *array) +{ + int i; + + for (i = 0; i < array->value_cnt; i++) + json_free_value(array->values[i]); + free(array->values); + free(array); +} + +static void json_free_pair(struct json_pair *pair) +{ + json_free_value(pair->value); + free(pair->name); + free(pair); +} + +static void json_free_value(struct json_value *value) +{ + switch (value->type) { + case JSON_TYPE_STRING: + free(value->string); + break; + case JSON_TYPE_OBJECT: + json_free_object(value->object); + break; + case JSON_TYPE_ARRAY: + json_free_array(value->array); + break; + } + free(value); +} + +static int json_array_add_value(struct json_array *array, struct json_value *value) +{ + struct json_value **values = realloc(array->values, + sizeof(struct json_value *) * (array->value_cnt + 1)); + + if (!values) + return ENOMEM; + values[array->value_cnt] = value; + array->value_cnt++; + array->values = values; + + value->parent_type = JSON_PARENT_TYPE_ARRAY; + value->parent_array = array; + return 0; +} + +static int json_object_add_pair(struct json_object *obj, struct json_pair *pair) +{ + struct json_pair **pairs = realloc(obj->pairs, + sizeof(struct json_pair *) * (obj->pair_cnt + 1)); + if (!pairs) + return ENOMEM; + pairs[obj->pair_cnt] = pair; + obj->pair_cnt++; + obj->pairs = pairs; + + pair->parent = obj; + return 0; +} + +int json_object_add_value_type(struct json_object *obj, const char *name, int type, ...) +{ + struct json_value *value; + struct json_pair *pair; + va_list args; + int ret; + + va_start(args, type); + if (type == JSON_TYPE_STRING) + value = json_create_value_string(va_arg(args, char *)); + else if (type == JSON_TYPE_INTEGER) + value = json_create_value_int(va_arg(args, long)); + else if (type == JSON_TYPE_FLOAT) + value = json_create_value_float(va_arg(args, double)); + else if (type == JSON_TYPE_OBJECT) + value = json_create_value_object(va_arg(args, struct json_object *)); + else + value = json_create_value_array(va_arg(args, struct json_array *)); + va_end(args); + + if (!value) + return ENOMEM; + + pair = json_create_pair(name, value); + if (!pair) { + json_free_value(value); + return ENOMEM; + } + ret = json_object_add_pair(obj, pair); + if (ret) { + json_free_pair(pair); + return ENOMEM; + } + return 0; +} + +static void json_print_array(struct json_array *array); +int json_array_add_value_type(struct json_array *array, int type, ...) +{ + struct json_value *value; + va_list args; + int ret; + + va_start(args, type); + if (type == JSON_TYPE_STRING) + value = json_create_value_string(va_arg(args, char *)); + else if (type == JSON_TYPE_INTEGER) + value = json_create_value_int(va_arg(args, long)); + else if (type == JSON_TYPE_FLOAT) + value = json_create_value_float(va_arg(args, double)); + else if (type == JSON_TYPE_OBJECT) + value = json_create_value_object(va_arg(args, struct json_object *)); + else + value = json_create_value_array(va_arg(args, struct json_array *)); + va_end(args); + + if (!value) + return ENOMEM; + + ret = json_array_add_value(array, value); + if (ret) { + json_free_value(value); + return ENOMEM; + } + return 0; +} + +static int json_value_level(struct json_value *value); +static int json_pair_level(struct json_pair *pair); +static int json_array_level(struct json_array *array); +static int json_object_level(struct json_object *object) +{ + if (object->parent == NULL) + return 0; + return json_value_level(object->parent); +} + +static int json_pair_level(struct json_pair *pair) +{ + return json_object_level(pair->parent) + 1; +} + +static int json_array_level(struct json_array *array) +{ + return json_value_level(array->parent); +} + +static int json_value_level(struct json_value *value) +{ + if (value->parent_type == JSON_PARENT_TYPE_PAIR) + return json_pair_level(value->parent_pair); + else + return json_array_level(value->parent_array) + 1; +} + +static void json_print_level(int level) +{ + while (level-- > 0) + log_info(" "); +} + +static void json_print_pair(struct json_pair *pair); +static void json_print_array(struct json_array *array); +static void json_print_value(struct json_value *value); +void json_print_object(struct json_object *obj) +{ + int i; + + log_info("{\n"); + for (i = 0; i < obj->pair_cnt; i++) { + if (i > 0) + log_info(",\n"); + json_print_pair(obj->pairs[i]); + } + log_info("\n"); + json_print_level(json_object_level(obj)); + log_info("}"); +} + +static void json_print_pair(struct json_pair *pair) +{ + json_print_level(json_pair_level(pair)); + log_info("\"%s\" : ", pair->name); + json_print_value(pair->value); +} + +static void json_print_array(struct json_array *array) +{ + int i; + + log_info("[\n"); + for (i = 0; i < array->value_cnt; i++) { + if (i > 0) + log_info(",\n"); + json_print_level(json_value_level(array->values[i])); + json_print_value(array->values[i]); + } + log_info("\n"); + json_print_level(json_array_level(array)); + log_info("]"); +} + +static void json_print_value(struct json_value *value) +{ + switch (value->type) { + case JSON_TYPE_STRING: + log_info("\"%s\"", value->string); + break; + case JSON_TYPE_INTEGER: + log_info("%ld", value->integer_number); + break; + case JSON_TYPE_FLOAT: + log_info("%.2f", value->float_number); + break; + case JSON_TYPE_OBJECT: + json_print_object(value->object); + break; + case JSON_TYPE_ARRAY: + json_print_array(value->array); + break; + } +} diff --git a/json.h b/json.h new file mode 100644 index 0000000..4d05e82 --- /dev/null +++ b/json.h @@ -0,0 +1,77 @@ +#ifndef __JSON__H +#define __JSON__H +struct json_object; +struct json_array; +struct json_pair; + +#define JSON_TYPE_STRING 0 +#define JSON_TYPE_INTEGER 1 +#define JSON_TYPE_FLOAT 2 +#define JSON_TYPE_OBJECT 3 +#define JSON_TYPE_ARRAY 4 +#define JSON_PARENT_TYPE_PAIR 0 +#define JSON_PARENT_TYPE_ARRAY 1 +struct json_value { + int type; + union { + long integer_number; + double float_number; + char *string; + struct json_object *object; + struct json_array *array; + }; + int parent_type; + union { + struct json_pair *parent_pair; + struct json_array *parent_array; + }; +}; + +struct json_array { + struct json_value **values; + int value_cnt; + struct json_value *parent; +}; + +struct json_object { + struct json_pair **pairs; + int pair_cnt; + struct json_value *parent; +}; + +struct json_pair { + char *name; + struct json_value *value; + struct json_object *parent; +}; + +struct json_object *json_create_object(void); +struct json_array *json_create_array(void); + +void json_free_object(struct json_object *obj); + +int json_object_add_value_type(struct json_object *obj, const char *name, int type, ...); +#define json_object_add_value_int(obj, name, val) \ + json_object_add_value_type((obj), name, JSON_TYPE_INTEGER, (val)) +#define json_object_add_value_float(obj, name, val) \ + json_object_add_value_type((obj), name, JSON_TYPE_FLOAT, (val)) +#define json_object_add_value_string(obj, name, val) \ + json_object_add_value_type((obj), name, JSON_TYPE_STRING, (val)) +#define json_object_add_value_object(obj, name, val) \ + json_object_add_value_type((obj), name, JSON_TYPE_OBJECT, (val)) +#define json_object_add_value_array(obj, name, val) \ + json_object_add_value_type((obj), name, JSON_TYPE_ARRAY, (val)) +int json_array_add_value_type(struct json_array *array, int type, ...); +#define json_array_add_value_int(obj, val) \ + json_array_add_value_type((obj), JSON_TYPE_INTEGER, (val)) +#define json_array_add_value_float(obj, val) \ + json_array_add_value_type((obj), JSON_TYPE_FLOAT, (val)) +#define json_array_add_value_string(obj, val) \ + json_array_add_value_type((obj), JSON_TYPE_STRING, (val)) +#define json_array_add_value_object(obj, val) \ + json_array_add_value_type((obj), JSON_TYPE_OBJECT, (val)) +#define json_array_add_value_array(obj, val) \ + json_array_add_value_type((obj), JSON_TYPE_ARRAY, (val)) + +void json_print_object(struct json_object *obj); +#endif diff --git a/stat.c b/stat.c index 4e00177..2665952 100644 --- a/stat.c +++ b/stat.c @@ -10,6 +10,7 @@ #include "fio.h" #include "diskutil.h" #include "lib/ieee754.h" +#include "json.h" void update_rusage_stat(struct thread_data *td) { @@ -678,6 +679,109 @@ static void show_ddir_status_terse(struct thread_stat *ts, log_info(";%lu;%lu;%f%%;%f;%f", 0UL, 0UL, 0.0, 0.0, 0.0); } +static void add_ddir_status_json(struct thread_stat *ts, + struct group_run_stats *rs, int ddir, struct json_object *parent) +{ + unsigned long min, max; + unsigned long long bw, iops; + unsigned int *ovals = NULL; + double mean, dev; + unsigned int len, minv, maxv; + int i; + const char *ddirname[] = {"read", "write", "trim"}; + struct json_object *dir_object, *tmp_object, *percentile_object; + char buf[120]; + double p_of_agg = 100.0; + + assert(ddir_rw(ddir)); + + dir_object = json_create_object(); + json_object_add_value_object(parent, ddirname[ddir], dir_object); + + iops = bw = 0; + if (ts->runtime[ddir]) { + uint64_t runt = ts->runtime[ddir]; + + bw = ((1000 * ts->io_bytes[ddir]) / runt) / 1024; + iops = (1000 * (uint64_t) ts->total_io_u[ddir]) / runt; + } + + json_object_add_value_int(dir_object, "io_bytes", ts->io_bytes[ddir] >> 10); + json_object_add_value_int(dir_object, "bw", bw); + json_object_add_value_int(dir_object, "iops", iops); + json_object_add_value_int(dir_object, "runtime", ts->runtime[ddir]); + + if (!calc_lat(&ts->slat_stat[ddir], &min, &max, &mean, &dev)) { + min = max = 0; + mean = dev = 0.0; + } + tmp_object = json_create_object(); + json_object_add_value_object(dir_object, "slat", tmp_object); + json_object_add_value_int(tmp_object, "min", min); + json_object_add_value_int(tmp_object, "max", max); + json_object_add_value_float(tmp_object, "mean", mean); + json_object_add_value_float(tmp_object, "stddev", dev); + + if (!calc_lat(&ts->clat_stat[ddir], &min, &max, &mean, &dev)) { + min = max = 0; + mean = dev = 0.0; + } + tmp_object = json_create_object(); + json_object_add_value_object(dir_object, "clat", tmp_object); + json_object_add_value_int(tmp_object, "min", min); + json_object_add_value_int(tmp_object, "max", max); + json_object_add_value_float(tmp_object, "mean", mean); + json_object_add_value_float(tmp_object, "stddev", dev); + + if (ts->clat_percentiles) { + len = calc_clat_percentiles(ts->io_u_plat[ddir], + ts->clat_stat[ddir].samples, + ts->percentile_list, &ovals, &maxv, + &minv); + } else + len = 0; + + percentile_object = json_create_object(); + json_object_add_value_object(tmp_object, "percentile", percentile_object); + for (i = 0; i < FIO_IO_U_LIST_MAX_LEN; i++) { + if (i >= len) { + json_object_add_value_int(percentile_object, "0.00", 0); + continue; + } + snprintf(buf, sizeof(buf) - 1, "%2.2f", ts->percentile_list[i].u.f); + json_object_add_value_int(percentile_object, (const char *)buf, ovals[i]); + } + + if (!calc_lat(&ts->lat_stat[ddir], &min, &max, &mean, &dev)) { + min = max = 0; + mean = dev = 0.0; + } + tmp_object = json_create_object(); + json_object_add_value_object(dir_object, "lat", tmp_object); + json_object_add_value_int(tmp_object, "min", min); + json_object_add_value_int(tmp_object, "max", max); + json_object_add_value_float(tmp_object, "mean", mean); + json_object_add_value_float(tmp_object, "stddev", dev); + if (ovals) + free(ovals); + + if (!calc_lat(&ts->bw_stat[ddir], &min, &max, &mean, &dev)) { + if (rs->agg[ddir]) { + p_of_agg = mean * 100 / (double) rs->agg[ddir]; + if (p_of_agg > 100.0) + p_of_agg = 100.0; + } + } else { + min = max = 0; + p_of_agg = mean = dev = 0.0; + } + json_object_add_value_int(dir_object, "bw_min", min); + json_object_add_value_int(dir_object, "bw_max", max); + json_object_add_value_float(dir_object, "bw_agg", mean); + json_object_add_value_float(dir_object, "bw_mean", mean); + json_object_add_value_float(dir_object, "bw_dev", dev); +} + static void show_thread_status_terse_v2(struct thread_stat *ts, struct group_run_stats *rs) { @@ -748,7 +852,7 @@ static void show_thread_status_terse_v3_v4(struct thread_stat *ts, int i; /* General Info */ - log_info("%s;%s;%s;%d;%d", ver, fio_version_string, + log_info("%d;%s;%s;%d;%d", ver, fio_version_string, ts->name, ts->groupid, ts->error); /* Log Read Status */ show_ddir_status_terse(ts, rs, DDIR_READ); @@ -790,7 +894,7 @@ static void show_thread_status_terse_v3_v4(struct thread_stat *ts, log_info(";%3.2f%%", io_u_lat_m[i]); /* disk util stats, if any */ - show_disk_util(1); + show_disk_util(1, NULL); /* Additional output if continue_on_error set - default off*/ if (ts->continue_on_error) @@ -803,6 +907,90 @@ static void show_thread_status_terse_v3_v4(struct thread_stat *ts, log_info("\n"); } +static struct json_object *show_thread_status_json(struct thread_stat *ts, + struct group_run_stats *rs) +{ + struct json_object *root, *tmp; + double io_u_dist[FIO_IO_U_MAP_NR]; + double io_u_lat_u[FIO_IO_U_LAT_U_NR]; + double io_u_lat_m[FIO_IO_U_LAT_M_NR]; + double usr_cpu, sys_cpu; + int i; + + root = json_create_object(); + json_object_add_value_string(root, "jobname", ts->name); + json_object_add_value_int(root, "groupid", ts->groupid); + json_object_add_value_int(root, "error", ts->error); + + add_ddir_status_json(ts, rs, DDIR_READ, root); + add_ddir_status_json(ts, rs, DDIR_WRITE, root); + add_ddir_status_json(ts, rs, DDIR_TRIM, root); + + /* CPU Usage */ + if (ts->total_run_time) { + double runt = (double) ts->total_run_time; + + usr_cpu = (double) ts->usr_time * 100 / runt; + sys_cpu = (double) ts->sys_time * 100 / runt; + } else { + usr_cpu = 0; + sys_cpu = 0; + } + json_object_add_value_float(root, "usr_cpu", usr_cpu); + json_object_add_value_float(root, "sys_cpu", sys_cpu); + json_object_add_value_int(root, "ctx", ts->ctx); + json_object_add_value_int(root, "majf", ts->majf); + json_object_add_value_int(root, "minf", ts->minf); + + + /* Calc % distribution of IO depths, usecond, msecond latency */ + stat_calc_dist(ts->io_u_map, ts_total_io_u(ts), io_u_dist); + stat_calc_lat_u(ts, io_u_lat_u); + stat_calc_lat_m(ts, io_u_lat_m); + + tmp = json_create_object(); + json_object_add_value_object(root, "iodepth_level", tmp); + /* Only show fixed 7 I/O depth levels*/ + for (i = 0; i < 7; i++) { + char name[20]; + if (i < 6) + snprintf(name, 19, "%d", 1 << i); + else + snprintf(name, 19, ">=%d", 1 << i); + json_object_add_value_float(tmp, (const char *)name, io_u_dist[i]); + } + + tmp = json_create_object(); + json_object_add_value_object(root, "latency_us", tmp); + /* Microsecond latency */ + for (i = 0; i < FIO_IO_U_LAT_U_NR; i++) { + const char *ranges[] = { "2", "4", "10", "20", "50", "100", + "250", "500", "750", "1000", }; + json_object_add_value_float(tmp, ranges[i], io_u_lat_u[i]); + } + /* Millisecond latency */ + tmp = json_create_object(); + json_object_add_value_object(root, "latency_ms", tmp); + for (i = 0; i < FIO_IO_U_LAT_M_NR; i++) { + const char *ranges[] = { "2", "4", "10", "20", "50", "100", + "250", "500", "750", "1000", "2000", + ">=2000", }; + json_object_add_value_float(tmp, ranges[i], io_u_lat_m[i]); + } + + /* Additional output if continue_on_error set - default off*/ + if (ts->continue_on_error) { + json_object_add_value_int(root, "total_err", ts->total_err_count); + json_object_add_value_int(root, "total_err", ts->first_error); + } + + /* Additional output if description is set */ + if (strlen(ts->description)) + json_object_add_value_string(root, "desc", ts->description); + + return root; +} + static void show_thread_status_terse(struct thread_stat *ts, struct group_run_stats *rs) { @@ -949,6 +1137,8 @@ void show_run_stats(void) struct thread_stat *threadstats, *ts; int i, j, nr_ts, last_ts, idx; int kb_base_warned = 0; + struct json_object *root = NULL; + struct json_array *array = NULL; runstats = malloc(sizeof(struct group_run_stats) * (groupid + 1)); @@ -1090,8 +1280,14 @@ void show_run_stats(void) /* * don't overwrite last signal output */ - if (!terse_output) + if (output_format == FIO_OUTPUT_NORMAL) log_info("\n"); + else if (output_format == FIO_OUTPUT_JSON) { + root = json_create_object(); + json_object_add_value_string(root, "fio version", fio_version_string); + array = json_create_array(); + json_object_add_value_array(root, "jobs", array); + } for (i = 0; i < nr_ts; i++) { ts = &threadstats[i]; @@ -1099,11 +1295,22 @@ void show_run_stats(void) if (is_backend) fio_server_send_ts(ts, rs); - else if (terse_output) + else if (output_format == FIO_OUTPUT_TERSE) show_thread_status_terse(ts, rs); - else + else if (output_format == FIO_OUTPUT_JSON) { + struct json_object *tmp = show_thread_status_json(ts, rs); + json_array_add_value_object(array, tmp); + } else show_thread_status(ts, rs); } + if (output_format == FIO_OUTPUT_JSON) { + /* disk util stats, if any */ + show_disk_util(1, root); + + json_print_object(root); + log_info("\n"); + json_free_object(root); + } for (i = 0; i < groupid + 1; i++) { rs = &runstats[i]; @@ -1111,14 +1318,14 @@ void show_run_stats(void) rs->groupid = i; if (is_backend) fio_server_send_gs(rs); - else if (!terse_output) + else if (output_format == FIO_OUTPUT_NORMAL) show_group_stats(rs); } if (is_backend) fio_server_send_du(); - else if (!terse_output) - show_disk_util(0); + else if (output_format == FIO_OUTPUT_NORMAL) + show_disk_util(0, NULL); free(runstats); free(threadstats); -- To unsubscribe from this list: send the line "unsubscribe fio" in the body of a message to majordomo@xxxxxxxxxxxxxxx More majordomo info at http://vger.kernel.org/majordomo-info.html