2 * gfio - gui front end for fio - the flexible io tester
4 * Copyright (C) 2012 Stephen M. Cameron <stephenmcameron@gmail.com>
5 * Copyright (C) 2012 Jens Axboe <axboe@kernel.dk>
7 * The license below covers all files distributed with fio unless otherwise
8 * noted in the file itself.
10 * This program is free software; you can redistribute it and/or modify
11 * it under the terms of the GNU General Public License version 2 as
12 * published by the Free Software Foundation.
14 * This program is distributed in the hope that it will be useful,
15 * but WITHOUT ANY WARRANTY; without even the implied warranty of
16 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
17 * GNU General Public License for more details.
19 * You should have received a copy of the GNU General Public License
20 * along with this program; if not, write to the Free Software
21 * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
34 static int gfio_server_running;
35 static const char *gfio_graph_font;
37 static void gfio_update_thread_status(char *status_message, double perc);
39 #define ARRAYSIZE(x) (sizeof((x)) / (sizeof((x)[0])))
41 typedef void (*clickfunction)(GtkWidget *widget, gpointer data);
43 static void connect_clicked(GtkWidget *widget, gpointer data);
44 static void start_job_clicked(GtkWidget *widget, gpointer data);
46 static struct button_spec {
47 const char *buttontext;
49 const char *tooltiptext;
50 const int start_insensitive;
51 } buttonspeclist[] = {
52 #define CONNECT_BUTTON 0
53 #define START_JOB_BUTTON 1
54 { "Connect", connect_clicked, "Connect to host", 0 },
57 "Send current fio job to fio server to be executed", 1 },
79 GtkWidget *write_iops;
89 GtkWidget *bottomalign;
90 GtkWidget *thread_status_pb;
92 GtkWidget *button[ARRAYSIZE(buttonspeclist)];
93 GtkWidget *scrolled_window;
94 #define DRAWING_AREA_XDIM 1000
95 #define DRAWING_AREA_YDIM 400
96 GtkWidget *drawing_area;
97 int drawing_area_xdim;
98 int drawing_area_ydim;
99 GtkWidget *error_info_bar;
100 GtkWidget *error_label;
101 GtkWidget *results_notebook;
102 GtkWidget *results_window;
103 GtkListStore *log_model;
107 struct probe_widget probe;
108 struct eta_widget eta;
113 struct graph *iops_graph;
114 struct graph *bandwidth_graph;
115 struct fio_client *client;
122 GtkWidget *results_widget;
123 GtkWidget *disk_util_frame;
126 static void setup_iops_graph(struct gui *ui)
129 graph_free(ui->iops_graph);
130 ui->iops_graph = graph_new(DRAWING_AREA_XDIM / 2.0,
131 DRAWING_AREA_YDIM, gfio_graph_font);
132 graph_title(ui->iops_graph, "IOPS");
133 graph_x_title(ui->iops_graph, "Time (secs)");
134 graph_add_label(ui->iops_graph, "Read IOPS");
135 graph_add_label(ui->iops_graph, "Write IOPS");
136 graph_set_color(ui->iops_graph, "Read IOPS", 0.13, 0.54, 0.13);
137 graph_set_color(ui->iops_graph, "Write IOPS", 1.0, 0.0, 0.0);
138 line_graph_set_data_count_limit(ui->iops_graph, 100);
141 static void setup_bandwidth_graph(struct gui *ui)
143 if (ui->bandwidth_graph)
144 graph_free(ui->bandwidth_graph);
145 ui->bandwidth_graph = graph_new(DRAWING_AREA_XDIM / 2.0,
146 DRAWING_AREA_YDIM, gfio_graph_font);
147 graph_title(ui->bandwidth_graph, "Bandwidth");
148 graph_x_title(ui->bandwidth_graph, "Time (secs)");
149 graph_add_label(ui->bandwidth_graph, "Read Bandwidth");
150 graph_add_label(ui->bandwidth_graph, "Write Bandwidth");
151 graph_set_color(ui->bandwidth_graph, "Read Bandwidth", 0.13, 0.54, 0.13);
152 graph_set_color(ui->bandwidth_graph, "Write Bandwidth", 1.0, 0.0, 0.0);
153 line_graph_set_data_count_limit(ui->bandwidth_graph, 100);
156 static void clear_ui_info(struct gui *ui)
158 gtk_label_set_text(GTK_LABEL(ui->probe.hostname), "");
159 gtk_label_set_text(GTK_LABEL(ui->probe.os), "");
160 gtk_label_set_text(GTK_LABEL(ui->probe.arch), "");
161 gtk_label_set_text(GTK_LABEL(ui->probe.fio_ver), "");
162 gtk_entry_set_text(GTK_ENTRY(ui->eta.name), "");
163 gtk_entry_set_text(GTK_ENTRY(ui->eta.iotype), "");
164 gtk_entry_set_text(GTK_ENTRY(ui->eta.ioengine), "");
165 gtk_entry_set_text(GTK_ENTRY(ui->eta.iodepth), "");
166 gtk_entry_set_text(GTK_ENTRY(ui->eta.jobs), "");
167 gtk_entry_set_text(GTK_ENTRY(ui->eta.files), "");
168 gtk_entry_set_text(GTK_ENTRY(ui->eta.read_bw), "");
169 gtk_entry_set_text(GTK_ENTRY(ui->eta.read_iops), "");
170 gtk_entry_set_text(GTK_ENTRY(ui->eta.write_bw), "");
171 gtk_entry_set_text(GTK_ENTRY(ui->eta.write_iops), "");
174 static GtkWidget *new_info_entry_in_frame(GtkWidget *box, const char *label)
176 GtkWidget *entry, *frame;
178 frame = gtk_frame_new(label);
179 entry = gtk_entry_new();
180 gtk_entry_set_editable(GTK_ENTRY(entry), 0);
181 gtk_box_pack_start(GTK_BOX(box), frame, TRUE, TRUE, 3);
182 gtk_container_add(GTK_CONTAINER(frame), entry);
187 static GtkWidget *new_info_label_in_frame(GtkWidget *box, const char *label)
189 GtkWidget *label_widget;
192 frame = gtk_frame_new(label);
193 label_widget = gtk_label_new(NULL);
194 gtk_box_pack_start(GTK_BOX(box), frame, TRUE, TRUE, 3);
195 gtk_container_add(GTK_CONTAINER(frame), label_widget);
200 static GtkWidget *create_spinbutton(GtkWidget *hbox, double min, double max, double defval)
202 GtkWidget *button, *box;
204 box = gtk_hbox_new(FALSE, 3);
205 gtk_container_add(GTK_CONTAINER(hbox), box);
207 button = gtk_spin_button_new_with_range(min, max, 1.0);
208 gtk_box_pack_start(GTK_BOX(box), button, TRUE, TRUE, 0);
210 gtk_spin_button_set_update_policy(GTK_SPIN_BUTTON(button), GTK_UPDATE_IF_VALID);
211 gtk_spin_button_set_value(GTK_SPIN_BUTTON(button), defval);
216 static void gfio_set_connected(struct gui *ui, int connected)
219 gtk_widget_set_sensitive(ui->button[START_JOB_BUTTON], 1);
221 gtk_button_set_label(GTK_BUTTON(ui->button[CONNECT_BUTTON]), "Disconnect");
222 gtk_widget_set_sensitive(ui->button[CONNECT_BUTTON], 1);
225 gtk_button_set_label(GTK_BUTTON(ui->button[CONNECT_BUTTON]), "Connect");
226 gtk_widget_set_sensitive(ui->button[START_JOB_BUTTON], 0);
227 gtk_widget_set_sensitive(ui->button[CONNECT_BUTTON], 1);
231 static void label_set_int_value(GtkWidget *entry, unsigned int val)
235 sprintf(tmp, "%u", val);
236 gtk_label_set_text(GTK_LABEL(entry), tmp);
239 static void entry_set_int_value(GtkWidget *entry, unsigned int val)
243 sprintf(tmp, "%u", val);
244 gtk_entry_set_text(GTK_ENTRY(entry), tmp);
248 #define ALIGN_RIGHT 2
252 GtkTreeViewColumn *tree_view_column(GtkWidget *tree_view, int index, const char *title, unsigned int flags)
254 GtkCellRenderer *renderer;
255 GtkTreeViewColumn *col;
256 double xalign = 0.0; /* left as default */
257 PangoAlignment align;
260 align = (flags & ALIGN_LEFT) ? PANGO_ALIGN_LEFT :
261 (flags & ALIGN_RIGHT) ? PANGO_ALIGN_RIGHT :
263 visible = !(flags & INVISIBLE);
265 renderer = gtk_cell_renderer_text_new();
266 col = gtk_tree_view_column_new();
268 gtk_tree_view_column_set_title(col, title);
269 if (!(flags & UNSORTABLE))
270 gtk_tree_view_column_set_sort_column_id(col, index);
271 gtk_tree_view_column_set_resizable(col, TRUE);
272 gtk_tree_view_column_pack_start(col, renderer, TRUE);
273 gtk_tree_view_column_add_attribute(col, renderer, "text", index);
274 gtk_object_set(GTK_OBJECT(renderer), "alignment", align, NULL);
276 case PANGO_ALIGN_LEFT:
279 case PANGO_ALIGN_CENTER:
282 case PANGO_ALIGN_RIGHT:
286 gtk_cell_renderer_set_alignment(GTK_CELL_RENDERER(renderer), xalign, 0.5);
287 gtk_tree_view_column_set_visible(col, visible);
288 gtk_tree_view_append_column(GTK_TREE_VIEW(tree_view), col);
292 static void gfio_ui_setup_log(struct gui *ui)
294 GtkTreeSelection *selection;
296 GtkWidget *tree_view;
298 model = gtk_list_store_new(4, G_TYPE_STRING, G_TYPE_STRING, G_TYPE_INT, G_TYPE_STRING);
300 tree_view = gtk_tree_view_new_with_model(GTK_TREE_MODEL(model));
301 gtk_widget_set_can_focus(tree_view, FALSE);
303 selection = gtk_tree_view_get_selection(GTK_TREE_VIEW(tree_view));
304 gtk_tree_selection_set_mode(GTK_TREE_SELECTION(selection), GTK_SELECTION_BROWSE);
305 g_object_set(G_OBJECT(tree_view), "headers-visible", TRUE,
306 "enable-grid-lines", GTK_TREE_VIEW_GRID_LINES_BOTH, NULL);
308 tree_view_column(tree_view, 0, "Time", ALIGN_RIGHT | UNSORTABLE);
309 tree_view_column(tree_view, 1, "Host", ALIGN_RIGHT | UNSORTABLE);
310 tree_view_column(tree_view, 2, "Level", ALIGN_RIGHT | UNSORTABLE);
311 tree_view_column(tree_view, 3, "Text", ALIGN_LEFT | UNSORTABLE);
313 ui->log_model = model;
314 ui->log_tree = tree_view;
317 static GtkWidget *gfio_output_clat_percentiles(unsigned int *ovals,
323 GType types[FIO_IO_U_LIST_MAX_LEN];
324 GtkWidget *tree_view;
325 GtkTreeSelection *selection;
330 for (i = 0; i < len; i++)
331 types[i] = G_TYPE_INT;
333 model = gtk_list_store_newv(len, types);
335 tree_view = gtk_tree_view_new_with_model(GTK_TREE_MODEL(model));
336 gtk_widget_set_can_focus(tree_view, FALSE);
338 g_object_set(G_OBJECT(tree_view), "headers-visible", TRUE,
339 "enable-grid-lines", GTK_TREE_VIEW_GRID_LINES_BOTH, NULL);
341 selection = gtk_tree_view_get_selection(GTK_TREE_VIEW(tree_view));
342 gtk_tree_selection_set_mode(GTK_TREE_SELECTION(selection), GTK_SELECTION_BROWSE);
344 for (i = 0; i < len; i++) {
347 sprintf(fbuf, "%2.2f%%", plist[i].u.f);
348 tree_view_column(tree_view, i, fbuf, ALIGN_RIGHT | UNSORTABLE);
351 gtk_list_store_append(model, &iter);
353 for (i = 0; i < len; i++) {
355 ovals[i] = (ovals[i] + 999) / 1000;
356 gtk_list_store_set(model, &iter, i, ovals[i], -1);
362 static void gfio_show_clat_percentiles(GtkWidget *vbox, struct thread_stat *ts,
365 unsigned int *io_u_plat = ts->io_u_plat[ddir];
366 unsigned long nr = ts->clat_stat[ddir].samples;
367 fio_fp64_t *plist = ts->percentile_list;
368 unsigned int *ovals, len, minv, maxv, scale_down;
370 GtkWidget *tree_view, *frame, *hbox;
373 len = calc_clat_percentiles(io_u_plat, nr, plist, &ovals, &maxv, &minv);
378 * We default to usecs, but if the value range is such that we
379 * should scale down to msecs, do that.
381 if (minv > 2000 && maxv > 99999) {
389 tree_view = gfio_output_clat_percentiles(ovals, plist, len, base, scale_down);
391 sprintf(tmp, "Completion percentiles (%s)", base);
392 frame = gtk_frame_new(tmp);
393 gtk_box_pack_start(GTK_BOX(vbox), frame, FALSE, FALSE, 5);
395 hbox = gtk_hbox_new(FALSE, 3);
396 gtk_container_add(GTK_CONTAINER(frame), hbox);
398 gtk_box_pack_start(GTK_BOX(hbox), tree_view, TRUE, FALSE, 3);
404 static void gfio_show_lat(GtkWidget *vbox, const char *name, unsigned long min,
405 unsigned long max, double mean, double dev)
407 const char *base = "(usec)";
408 GtkWidget *hbox, *label, *frame;
412 if (!usec_to_msec(&min, &max, &mean, &dev))
415 minp = num2str(min, 6, 1, 0);
416 maxp = num2str(max, 6, 1, 0);
418 sprintf(tmp, "%s %s", name, base);
419 frame = gtk_frame_new(tmp);
420 gtk_box_pack_start(GTK_BOX(vbox), frame, FALSE, FALSE, 5);
422 hbox = gtk_hbox_new(FALSE, 3);
423 gtk_container_add(GTK_CONTAINER(frame), hbox);
425 label = new_info_label_in_frame(hbox, "Minimum");
426 gtk_label_set_text(GTK_LABEL(label), minp);
427 label = new_info_label_in_frame(hbox, "Maximum");
428 gtk_label_set_text(GTK_LABEL(label), maxp);
429 label = new_info_label_in_frame(hbox, "Average");
430 sprintf(tmp, "%5.02f", mean);
431 gtk_label_set_text(GTK_LABEL(label), tmp);
432 label = new_info_label_in_frame(hbox, "Standard deviation");
433 sprintf(tmp, "%5.02f", dev);
434 gtk_label_set_text(GTK_LABEL(label), tmp);
445 static void gfio_show_ddir_status(GtkWidget *mbox, struct group_run_stats *rs,
446 struct thread_stat *ts, int ddir)
448 const char *ddir_label[2] = { "Read", "Write" };
449 GtkWidget *frame, *label, *box, *vbox, *main_vbox;
450 unsigned long min[3], max[3], runt;
451 unsigned long long bw, iops;
452 unsigned int flags = 0;
453 double mean[3], dev[3];
454 char *io_p, *bw_p, *iops_p;
457 if (!ts->runtime[ddir])
460 i2p = is_power_of_2(rs->kb_base);
461 runt = ts->runtime[ddir];
463 bw = (1000 * ts->io_bytes[ddir]) / runt;
464 io_p = num2str(ts->io_bytes[ddir], 6, 1, i2p);
465 bw_p = num2str(bw, 6, 1, i2p);
467 iops = (1000 * (uint64_t)ts->total_io_u[ddir]) / runt;
468 iops_p = num2str(iops, 6, 1, 0);
470 box = gtk_hbox_new(FALSE, 3);
471 gtk_box_pack_start(GTK_BOX(mbox), box, TRUE, FALSE, 3);
473 frame = gtk_frame_new(ddir_label[ddir]);
474 gtk_box_pack_start(GTK_BOX(box), frame, FALSE, FALSE, 5);
476 main_vbox = gtk_vbox_new(FALSE, 3);
477 gtk_container_add(GTK_CONTAINER(frame), main_vbox);
479 box = gtk_hbox_new(FALSE, 3);
480 gtk_box_pack_start(GTK_BOX(main_vbox), box, TRUE, FALSE, 3);
482 label = new_info_label_in_frame(box, "IO");
483 gtk_label_set_text(GTK_LABEL(label), io_p);
484 label = new_info_label_in_frame(box, "Bandwidth");
485 gtk_label_set_text(GTK_LABEL(label), bw_p);
486 label = new_info_label_in_frame(box, "IOPS");
487 gtk_label_set_text(GTK_LABEL(label), iops_p);
488 label = new_info_label_in_frame(box, "Runtime (msec)");
489 label_set_int_value(label, ts->runtime[ddir]);
491 if (calc_lat(&ts->bw_stat[ddir], &min[0], &max[0], &mean[0], &dev[0])) {
492 double p_of_agg = 100.0;
493 const char *bw_str = "KB";
497 p_of_agg = mean[0] * 100 / (double) rs->agg[ddir];
498 if (p_of_agg > 100.0)
502 if (mean[0] > 999999.9) {
510 sprintf(tmp, "Bandwidth (%s)", bw_str);
511 frame = gtk_frame_new(tmp);
512 gtk_box_pack_start(GTK_BOX(main_vbox), frame, FALSE, FALSE, 5);
514 box = gtk_hbox_new(FALSE, 3);
515 gtk_container_add(GTK_CONTAINER(frame), box);
517 label = new_info_label_in_frame(box, "Minimum");
518 label_set_int_value(label, min[0]);
519 label = new_info_label_in_frame(box, "Maximum");
520 label_set_int_value(label, max[0]);
521 label = new_info_label_in_frame(box, "Percentage of jobs");
522 sprintf(tmp, "%3.2f%%", p_of_agg);
523 gtk_label_set_text(GTK_LABEL(label), tmp);
524 label = new_info_label_in_frame(box, "Average");
525 sprintf(tmp, "%5.02f", mean[0]);
526 gtk_label_set_text(GTK_LABEL(label), tmp);
527 label = new_info_label_in_frame(box, "Standard deviation");
528 sprintf(tmp, "%5.02f", dev[0]);
529 gtk_label_set_text(GTK_LABEL(label), tmp);
532 if (calc_lat(&ts->slat_stat[ddir], &min[0], &max[0], &mean[0], &dev[0]))
534 if (calc_lat(&ts->clat_stat[ddir], &min[1], &max[1], &mean[1], &dev[1]))
536 if (calc_lat(&ts->lat_stat[ddir], &min[2], &max[2], &mean[2], &dev[2]))
540 frame = gtk_frame_new("Latency");
541 gtk_box_pack_start(GTK_BOX(main_vbox), frame, FALSE, FALSE, 5);
543 vbox = gtk_vbox_new(FALSE, 3);
544 gtk_container_add(GTK_CONTAINER(frame), vbox);
546 if (flags & GFIO_SLAT)
547 gfio_show_lat(vbox, "Submission latency", min[0], max[0], mean[0], dev[0]);
548 if (flags & GFIO_CLAT)
549 gfio_show_lat(vbox, "Completion latency", min[1], max[1], mean[1], dev[1]);
550 if (flags & GFIO_LAT)
551 gfio_show_lat(vbox, "Total latency", min[2], max[2], mean[2], dev[2]);
554 if (ts->clat_percentiles)
555 gfio_show_clat_percentiles(main_vbox, ts, ddir);
563 static GtkWidget *gfio_output_lat_buckets(double *lat, unsigned int num,
566 GtkWidget *tree_view;
567 GtkTreeSelection *selection;
574 * Check if all are empty, in which case don't bother
576 for (i = 0, skipped = 0; i < num; i++)
583 types = malloc(num * sizeof(GType));
585 for (i = 0; i < num; i++)
586 types[i] = G_TYPE_STRING;
588 model = gtk_list_store_newv(num, types);
592 tree_view = gtk_tree_view_new_with_model(GTK_TREE_MODEL(model));
593 gtk_widget_set_can_focus(tree_view, FALSE);
595 g_object_set(G_OBJECT(tree_view), "headers-visible", TRUE,
596 "enable-grid-lines", GTK_TREE_VIEW_GRID_LINES_BOTH, NULL);
598 selection = gtk_tree_view_get_selection(GTK_TREE_VIEW(tree_view));
599 gtk_tree_selection_set_mode(GTK_TREE_SELECTION(selection), GTK_SELECTION_BROWSE);
601 for (i = 0; i < num; i++)
602 tree_view_column(tree_view, i, labels[i], ALIGN_RIGHT | UNSORTABLE);
604 gtk_list_store_append(model, &iter);
606 for (i = 0; i < num; i++) {
610 sprintf(fbuf, "0.00");
612 sprintf(fbuf, "%3.2f%%", lat[i]);
614 gtk_list_store_set(model, &iter, i, fbuf, -1);
620 static void gfio_show_latency_buckets(GtkWidget *vbox, struct thread_stat *ts)
622 GtkWidget *box, *frame, *tree_view;
623 double io_u_lat_u[FIO_IO_U_LAT_U_NR];
624 double io_u_lat_m[FIO_IO_U_LAT_M_NR];
625 const char *uranges[] = { "2", "4", "10", "20", "50", "100",
626 "250", "500", "750", "1000", };
627 const char *mranges[] = { "2", "4", "10", "20", "50", "100",
628 "250", "500", "750", "1000", "2000",
631 stat_calc_lat_u(ts, io_u_lat_u);
632 stat_calc_lat_m(ts, io_u_lat_m);
634 tree_view = gfio_output_lat_buckets(io_u_lat_u, FIO_IO_U_LAT_U_NR, uranges);
636 frame = gtk_frame_new("Latency buckets (usec)");
637 gtk_box_pack_start(GTK_BOX(vbox), frame, FALSE, FALSE, 5);
639 box = gtk_hbox_new(FALSE, 3);
640 gtk_container_add(GTK_CONTAINER(frame), box);
641 gtk_box_pack_start(GTK_BOX(box), tree_view, TRUE, FALSE, 3);
644 tree_view = gfio_output_lat_buckets(io_u_lat_m, FIO_IO_U_LAT_M_NR, mranges);
646 frame = gtk_frame_new("Latency buckets (msec)");
647 gtk_box_pack_start(GTK_BOX(vbox), frame, FALSE, FALSE, 5);
649 box = gtk_hbox_new(FALSE, 3);
650 gtk_container_add(GTK_CONTAINER(frame), box);
651 gtk_box_pack_start(GTK_BOX(box), tree_view, TRUE, FALSE, 3);
655 static void gfio_show_cpu_usage(GtkWidget *vbox, struct thread_stat *ts)
657 GtkWidget *box, *frame, *entry;
658 double usr_cpu, sys_cpu;
659 unsigned long runtime;
662 runtime = ts->total_run_time;
664 double runt = (double) runtime;
666 usr_cpu = (double) ts->usr_time * 100 / runt;
667 sys_cpu = (double) ts->sys_time * 100 / runt;
673 frame = gtk_frame_new("OS resources");
674 gtk_box_pack_start(GTK_BOX(vbox), frame, FALSE, FALSE, 5);
676 box = gtk_hbox_new(FALSE, 3);
677 gtk_container_add(GTK_CONTAINER(frame), box);
679 entry = new_info_entry_in_frame(box, "User CPU");
680 sprintf(tmp, "%3.2f%%", usr_cpu);
681 gtk_entry_set_text(GTK_ENTRY(entry), tmp);
682 entry = new_info_entry_in_frame(box, "System CPU");
683 sprintf(tmp, "%3.2f%%", sys_cpu);
684 gtk_entry_set_text(GTK_ENTRY(entry), tmp);
685 entry = new_info_entry_in_frame(box, "Context switches");
686 entry_set_int_value(entry, ts->ctx);
687 entry = new_info_entry_in_frame(box, "Major faults");
688 entry_set_int_value(entry, ts->majf);
689 entry = new_info_entry_in_frame(box, "Minor faults");
690 entry_set_int_value(entry, ts->minf);
692 static void gfio_add_sc_depths_tree(GtkListStore *model,
693 struct thread_stat *ts, unsigned int len,
696 double io_u_dist[FIO_IO_U_MAP_NR];
698 /* Bits 0, and 3-8 */
699 const int add_mask = 0x1f9;
703 stat_calc_dist(ts->io_u_submit, ts->total_submit, io_u_dist);
705 stat_calc_dist(ts->io_u_complete, ts->total_complete, io_u_dist);
707 gtk_list_store_append(model, &iter);
709 gtk_list_store_set(model, &iter, 0, submit ? "Submit" : "Complete", -1);
711 for (i = 1, j = 0; i < len; i++) {
714 if (!(add_mask & (1UL << (i - 1))))
715 sprintf(fbuf, "0.0%%");
717 sprintf(fbuf, "%3.1f%%", io_u_dist[j]);
721 gtk_list_store_set(model, &iter, i, fbuf, -1);
726 static void gfio_add_total_depths_tree(GtkListStore *model,
727 struct thread_stat *ts, unsigned int len)
729 double io_u_dist[FIO_IO_U_MAP_NR];
731 /* Bits 1-6, and 8 */
732 const int add_mask = 0x17e;
735 stat_calc_dist(ts->io_u_map, ts_total_io_u(ts), io_u_dist);
737 gtk_list_store_append(model, &iter);
739 gtk_list_store_set(model, &iter, 0, "Total", -1);
741 for (i = 1, j = 0; i < len; i++) {
744 if (!(add_mask & (1UL << (i - 1))))
745 sprintf(fbuf, "0.0%%");
747 sprintf(fbuf, "%3.1f%%", io_u_dist[j]);
751 gtk_list_store_set(model, &iter, i, fbuf, -1);
756 static void gfio_show_io_depths(GtkWidget *vbox, struct thread_stat *ts)
758 GtkWidget *frame, *box, *tree_view;
759 GtkTreeSelection *selection;
761 GType types[FIO_IO_U_MAP_NR + 1];
764 const char *labels[NR_LABELS] = { "Depth", "0", "1", "2", "4", "8", "16", "32", "64", ">= 64" };
766 frame = gtk_frame_new("IO depths");
767 gtk_box_pack_start(GTK_BOX(vbox), frame, FALSE, FALSE, 5);
769 box = gtk_hbox_new(FALSE, 3);
770 gtk_container_add(GTK_CONTAINER(frame), box);
772 for (i = 0; i < NR_LABELS; i++)
773 types[i] = G_TYPE_STRING;
775 model = gtk_list_store_newv(NR_LABELS, types);
777 tree_view = gtk_tree_view_new_with_model(GTK_TREE_MODEL(model));
778 gtk_widget_set_can_focus(tree_view, FALSE);
780 g_object_set(G_OBJECT(tree_view), "headers-visible", TRUE,
781 "enable-grid-lines", GTK_TREE_VIEW_GRID_LINES_BOTH, NULL);
783 selection = gtk_tree_view_get_selection(GTK_TREE_VIEW(tree_view));
784 gtk_tree_selection_set_mode(GTK_TREE_SELECTION(selection), GTK_SELECTION_BROWSE);
786 for (i = 0; i < NR_LABELS; i++)
787 tree_view_column(tree_view, i, labels[i], ALIGN_RIGHT | UNSORTABLE);
789 gfio_add_total_depths_tree(model, ts, NR_LABELS);
790 gfio_add_sc_depths_tree(model, ts, NR_LABELS, 1);
791 gfio_add_sc_depths_tree(model, ts, NR_LABELS, 0);
793 gtk_box_pack_start(GTK_BOX(box), tree_view, TRUE, FALSE, 3);
796 static gboolean results_window_delete(GtkWidget *w, gpointer data)
798 struct gui *ui = (struct gui *) data;
800 gtk_widget_destroy(w);
801 ui->results_window = NULL;
802 ui->results_notebook = NULL;
806 static GtkWidget *get_results_window(struct gui *ui)
808 GtkWidget *win, *notebook;
810 if (ui->results_window)
811 return ui->results_notebook;
813 win = gtk_window_new(GTK_WINDOW_TOPLEVEL);
814 gtk_window_set_title(GTK_WINDOW(win), "Results");
815 gtk_window_set_default_size(GTK_WINDOW(win), 1024, 768);
816 g_signal_connect(win, "delete-event", G_CALLBACK(results_window_delete), ui);
817 g_signal_connect(win, "destroy", G_CALLBACK(results_window_delete), ui);
819 notebook = gtk_notebook_new();
820 gtk_container_add(GTK_CONTAINER(win), notebook);
822 ui->results_window = win;
823 ui->results_notebook = notebook;
824 return ui->results_notebook;
827 static void gfio_display_ts(struct fio_client *client, struct thread_stat *ts,
828 struct group_run_stats *rs)
830 GtkWidget *res_win, *box, *vbox, *entry, *scroll;
831 struct gfio_client *gc = client->client_data;
835 res_win = get_results_window(gc->ui);
837 scroll = gtk_scrolled_window_new(NULL, NULL);
838 gtk_container_set_border_width(GTK_CONTAINER(scroll), 5);
839 gtk_scrolled_window_set_policy(GTK_SCROLLED_WINDOW(scroll), GTK_POLICY_AUTOMATIC, GTK_POLICY_AUTOMATIC);
841 vbox = gtk_vbox_new(FALSE, 3);
843 box = gtk_hbox_new(FALSE, 0);
844 gtk_box_pack_start(GTK_BOX(vbox), box, TRUE, FALSE, 5);
846 gtk_scrolled_window_add_with_viewport(GTK_SCROLLED_WINDOW(scroll), vbox);
848 gtk_notebook_append_page(GTK_NOTEBOOK(res_win), scroll, gtk_label_new(ts->name));
850 gc->results_widget = vbox;
852 entry = new_info_entry_in_frame(box, "Name");
853 gtk_entry_set_text(GTK_ENTRY(entry), ts->name);
854 if (strlen(ts->description)) {
855 entry = new_info_entry_in_frame(box, "Description");
856 gtk_entry_set_text(GTK_ENTRY(entry), ts->description);
858 entry = new_info_entry_in_frame(box, "Group ID");
859 entry_set_int_value(entry, ts->groupid);
860 entry = new_info_entry_in_frame(box, "Jobs");
861 entry_set_int_value(entry, ts->members);
862 entry = new_info_entry_in_frame(box, "Error");
863 entry_set_int_value(entry, ts->error);
864 entry = new_info_entry_in_frame(box, "PID");
865 entry_set_int_value(entry, ts->pid);
867 if (ts->io_bytes[DDIR_READ])
868 gfio_show_ddir_status(vbox, rs, ts, DDIR_READ);
869 if (ts->io_bytes[DDIR_WRITE])
870 gfio_show_ddir_status(vbox, rs, ts, DDIR_WRITE);
872 gfio_show_latency_buckets(vbox, ts);
873 gfio_show_cpu_usage(vbox, ts);
874 gfio_show_io_depths(vbox, ts);
876 gtk_widget_show_all(gc->ui->results_window);
880 static void gfio_text_op(struct fio_client *client, struct fio_net_cmd *cmd)
882 struct cmd_text_pdu *p = (struct cmd_text_pdu *) cmd->payload;
883 struct gfio_client *gc = client->client_data;
887 char tmp[64], timebuf[80];
890 tm = localtime(&sec);
891 strftime(tmp, sizeof(tmp), "%Y-%m-%d %H:%M:%S", tm);
892 sprintf(timebuf, "%s.%03ld", tmp, p->log_usec / 1000);
896 gtk_list_store_append(gc->ui->log_model, &iter);
897 gtk_list_store_set(gc->ui->log_model, &iter, 0, timebuf, -1);
898 gtk_list_store_set(gc->ui->log_model, &iter, 1, client->hostname, -1);
899 gtk_list_store_set(gc->ui->log_model, &iter, 2, p->level, -1);
900 gtk_list_store_set(gc->ui->log_model, &iter, 3, p->buf, -1);
905 static void gfio_disk_util_op(struct fio_client *client, struct fio_net_cmd *cmd)
907 struct cmd_du_pdu *p = (struct cmd_du_pdu *) cmd->payload;
908 struct gfio_client *gc = client->client_data;
909 GtkWidget *box, *frame, *entry, *vbox;
915 if (!gc->results_widget)
918 if (!gc->disk_util_frame) {
919 gc->disk_util_frame = gtk_frame_new("Disk utilization");
920 gtk_box_pack_start(GTK_BOX(gc->results_widget), gc->disk_util_frame, FALSE, FALSE, 5);
923 vbox = gtk_vbox_new(FALSE, 3);
924 gtk_container_add(GTK_CONTAINER(gc->disk_util_frame), vbox);
926 frame = gtk_frame_new((char *) p->dus.name);
927 gtk_box_pack_start(GTK_BOX(vbox), frame, FALSE, FALSE, 2);
929 box = gtk_vbox_new(FALSE, 3);
930 gtk_container_add(GTK_CONTAINER(frame), box);
932 frame = gtk_frame_new("Read");
933 gtk_box_pack_start(GTK_BOX(box), frame, FALSE, FALSE, 2);
934 vbox = gtk_hbox_new(TRUE, 3);
935 gtk_container_add(GTK_CONTAINER(frame), vbox);
936 entry = new_info_entry_in_frame(vbox, "IOs");
937 entry_set_int_value(entry, p->dus.ios[0]);
938 entry = new_info_entry_in_frame(vbox, "Merges");
939 entry_set_int_value(entry, p->dus.merges[0]);
940 entry = new_info_entry_in_frame(vbox, "Sectors");
941 entry_set_int_value(entry, p->dus.sectors[0]);
942 entry = new_info_entry_in_frame(vbox, "Ticks");
943 entry_set_int_value(entry, p->dus.ticks[0]);
945 frame = gtk_frame_new("Write");
946 gtk_box_pack_start(GTK_BOX(box), frame, FALSE, FALSE, 2);
947 vbox = gtk_hbox_new(TRUE, 3);
948 gtk_container_add(GTK_CONTAINER(frame), vbox);
949 entry = new_info_entry_in_frame(vbox, "IOs");
950 entry_set_int_value(entry, p->dus.ios[1]);
951 entry = new_info_entry_in_frame(vbox, "Merges");
952 entry_set_int_value(entry, p->dus.merges[1]);
953 entry = new_info_entry_in_frame(vbox, "Sectors");
954 entry_set_int_value(entry, p->dus.sectors[1]);
955 entry = new_info_entry_in_frame(vbox, "Ticks");
956 entry_set_int_value(entry, p->dus.ticks[1]);
958 frame = gtk_frame_new("Shared");
959 gtk_box_pack_start(GTK_BOX(box), frame, FALSE, FALSE, 2);
960 vbox = gtk_hbox_new(TRUE, 3);
961 gtk_container_add(GTK_CONTAINER(frame), vbox);
962 entry = new_info_entry_in_frame(vbox, "IO ticks");
963 entry_set_int_value(entry, p->dus.io_ticks);
964 entry = new_info_entry_in_frame(vbox, "Time in queue");
965 entry_set_int_value(entry, p->dus.time_in_queue);
969 util = (double) 100 * p->dus.io_ticks / (double) p->dus.msec;
973 sprintf(tmp, "%3.2f%%", util);
974 entry = new_info_entry_in_frame(vbox, "Disk utilization");
975 gtk_entry_set_text(GTK_ENTRY(entry), tmp);
977 gtk_widget_show_all(gc->results_widget);
982 extern int sum_stat_clients;
983 extern struct thread_stat client_ts;
984 extern struct group_run_stats client_gs;
986 static int sum_stat_nr;
988 static void gfio_thread_status_op(struct fio_client *client,
989 struct fio_net_cmd *cmd)
991 struct cmd_ts_pdu *p = (struct cmd_ts_pdu *) cmd->payload;
993 gfio_display_ts(client, &p->ts, &p->rs);
995 if (sum_stat_clients == 1)
998 sum_thread_stats(&client_ts, &p->ts, sum_stat_nr);
999 sum_group_stats(&client_gs, &p->rs);
1001 client_ts.members++;
1002 client_ts.groupid = p->ts.groupid;
1004 if (++sum_stat_nr == sum_stat_clients) {
1005 strcpy(client_ts.name, "All clients");
1006 gfio_display_ts(client, &client_ts, &client_gs);
1010 static void gfio_group_stats_op(struct fio_client *client,
1011 struct fio_net_cmd *cmd)
1013 gdk_threads_enter();
1014 printf("gfio_group_stats_op called\n");
1015 fio_client_ops.group_stats(client, cmd);
1016 gdk_threads_leave();
1019 static gint on_config_drawing_area(GtkWidget *w, GdkEventConfigure *event)
1021 ui.drawing_area_xdim = w->allocation.width;
1022 ui.drawing_area_ydim = w->allocation.height;
1026 static int on_expose_drawing_area(GtkWidget *w, GdkEvent *event, gpointer p)
1028 struct gui *ui = (struct gui *) p;
1031 graph_set_size(ui->iops_graph, ui->drawing_area_xdim / 2.0,
1032 ui->drawing_area_ydim);
1033 graph_set_size(ui->bandwidth_graph, ui->drawing_area_xdim / 2.0,
1034 ui->drawing_area_ydim);
1035 cr = gdk_cairo_create(w->window);
1037 cairo_set_source_rgb(cr, 0, 0, 0);
1040 cairo_translate(cr, 0, 0);
1041 line_graph_draw(ui->bandwidth_graph, cr);
1046 cairo_translate(cr, ui->drawing_area_xdim / 2.0, 0);
1047 line_graph_draw(ui->iops_graph, cr);
1055 static void gfio_update_eta(struct jobs_eta *je)
1057 static int eta_good;
1064 gdk_threads_enter();
1069 if (je->eta_sec != INT_MAX && je->elapsed_sec) {
1070 perc = (double) je->elapsed_sec / (double) (je->elapsed_sec + je->eta_sec);
1071 eta_to_str(eta_str, je->eta_sec);
1074 sprintf(tmp, "%u", je->nr_running);
1075 gtk_entry_set_text(GTK_ENTRY(ui.eta.jobs), tmp);
1076 sprintf(tmp, "%u", je->files_open);
1077 gtk_entry_set_text(GTK_ENTRY(ui.eta.files), tmp);
1080 if (je->m_rate[0] || je->m_rate[1] || je->t_rate[0] || je->t_rate[1]) {
1081 if (je->m_rate || je->t_rate) {
1084 mr = num2str(je->m_rate, 4, 0, i2p);
1085 tr = num2str(je->t_rate, 4, 0, i2p);
1086 gtk_entry_set_text(GTK_ENTRY(ui.eta);
1087 p += sprintf(p, ", CR=%s/%s KB/s", tr, mr);
1090 } else if (je->m_iops || je->t_iops)
1091 p += sprintf(p, ", CR=%d/%d IOPS", je->t_iops, je->m_iops);
1093 gtk_entry_set_text(GTK_ENTRY(ui.eta.cr_bw), "---");
1094 gtk_entry_set_text(GTK_ENTRY(ui.eta.cr_iops), "---");
1095 gtk_entry_set_text(GTK_ENTRY(ui.eta.cw_bw), "---");
1096 gtk_entry_set_text(GTK_ENTRY(ui.eta.cw_iops), "---");
1099 if (je->eta_sec != INT_MAX && je->nr_running) {
1103 if ((!je->eta_sec && !eta_good) || je->nr_ramp == je->nr_running)
1104 strcpy(output, "-.-% done");
1108 sprintf(output, "%3.1f%% done", perc);
1111 rate_str[0] = num2str(je->rate[0], 5, 10, i2p);
1112 rate_str[1] = num2str(je->rate[1], 5, 10, i2p);
1114 iops_str[0] = num2str(je->iops[0], 4, 1, 0);
1115 iops_str[1] = num2str(je->iops[1], 4, 1, 0);
1117 gtk_entry_set_text(GTK_ENTRY(ui.eta.read_bw), rate_str[0]);
1118 gtk_entry_set_text(GTK_ENTRY(ui.eta.read_iops), iops_str[0]);
1119 gtk_entry_set_text(GTK_ENTRY(ui.eta.write_bw), rate_str[1]);
1120 gtk_entry_set_text(GTK_ENTRY(ui.eta.write_iops), iops_str[1]);
1122 graph_add_xy_data(ui.iops_graph, "Read IOPS", je->elapsed_sec, je->iops[0]);
1123 graph_add_xy_data(ui.iops_graph, "Write IOPS", je->elapsed_sec, je->iops[1]);
1124 graph_add_xy_data(ui.bandwidth_graph, "Read Bandwidth", je->elapsed_sec, je->rate[0]);
1125 graph_add_xy_data(ui.bandwidth_graph, "Write Bandwidth", je->elapsed_sec, je->rate[1]);
1134 char *dst = output + strlen(output);
1136 sprintf(dst, " - %s", eta_str);
1139 gfio_update_thread_status(output, perc);
1140 gdk_threads_leave();
1143 static void gfio_probe_op(struct fio_client *client, struct fio_net_cmd *cmd)
1145 struct cmd_probe_pdu *probe = (struct cmd_probe_pdu *) cmd->payload;
1146 struct gfio_client *gc = client->client_data;
1147 struct gui *ui = gc->ui;
1148 const char *os, *arch;
1151 os = fio_get_os_string(probe->os);
1155 arch = fio_get_arch_string(probe->arch);
1160 client->name = strdup((char *) probe->hostname);
1162 gdk_threads_enter();
1164 gtk_label_set_text(GTK_LABEL(ui->probe.hostname), (char *) probe->hostname);
1165 gtk_label_set_text(GTK_LABEL(ui->probe.os), os);
1166 gtk_label_set_text(GTK_LABEL(ui->probe.arch), arch);
1167 sprintf(buf, "%u.%u.%u", probe->fio_major, probe->fio_minor, probe->fio_patch);
1168 gtk_label_set_text(GTK_LABEL(ui->probe.fio_ver), buf);
1170 gfio_set_connected(ui, 1);
1172 gdk_threads_leave();
1175 static void gfio_update_thread_status(char *status_message, double perc)
1177 static char message[100];
1178 const char *m = message;
1180 strncpy(message, status_message, sizeof(message) - 1);
1181 gtk_progress_bar_set_text(
1182 GTK_PROGRESS_BAR(ui.thread_status_pb), m);
1183 gtk_progress_bar_set_fraction(
1184 GTK_PROGRESS_BAR(ui.thread_status_pb), perc / 100.0);
1185 gtk_widget_queue_draw(ui.window);
1188 static void gfio_quit_op(struct fio_client *client)
1190 struct gfio_client *gc = client->client_data;
1192 gdk_threads_enter();
1193 gfio_set_connected(gc->ui, 0);
1194 gdk_threads_leave();
1197 static void gfio_add_job_op(struct fio_client *client, struct fio_net_cmd *cmd)
1199 struct cmd_add_job_pdu *p = (struct cmd_add_job_pdu *) cmd->payload;
1200 struct gfio_client *gc = client->client_data;
1201 struct gui *ui = gc->ui;
1205 p->iodepth = le32_to_cpu(p->iodepth);
1206 p->rw = le32_to_cpu(p->rw);
1208 for (i = 0; i < 2; i++) {
1209 p->min_bs[i] = le32_to_cpu(p->min_bs[i]);
1210 p->max_bs[i] = le32_to_cpu(p->max_bs[i]);
1213 p->numjobs = le32_to_cpu(p->numjobs);
1214 p->group_reporting = le32_to_cpu(p->group_reporting);
1216 gdk_threads_enter();
1218 gtk_entry_set_text(GTK_ENTRY(ui->eta.name), (gchar *) p->jobname);
1219 gtk_entry_set_text(GTK_ENTRY(ui->eta.iotype), ddir_str(p->rw));
1220 gtk_entry_set_text(GTK_ENTRY(ui->eta.ioengine), (gchar *) p->ioengine);
1222 sprintf(tmp, "%u", p->iodepth);
1223 gtk_entry_set_text(GTK_ENTRY(ui->eta.iodepth), tmp);
1225 gdk_threads_leave();
1228 static void gfio_client_timed_out(struct fio_client *client)
1230 struct gfio_client *gc = client->client_data;
1231 GtkWidget *dialog, *label, *content;
1234 gdk_threads_enter();
1236 gfio_set_connected(gc->ui, 0);
1237 clear_ui_info(gc->ui);
1239 sprintf(buf, "Client %s: timeout talking to server.\n", client->hostname);
1241 dialog = gtk_dialog_new_with_buttons("Timed out!",
1242 GTK_WINDOW(gc->ui->window),
1243 GTK_DIALOG_MODAL | GTK_DIALOG_DESTROY_WITH_PARENT,
1244 GTK_STOCK_OK, GTK_RESPONSE_OK, NULL);
1246 /* gtk_dialog_get_content_area() is 2.14 and newer */
1247 content = GTK_DIALOG(dialog)->vbox;
1249 label = gtk_label_new((const gchar *) buf);
1250 gtk_container_add(GTK_CONTAINER(content), label);
1251 gtk_widget_show_all(dialog);
1252 gtk_dialog_set_default_response(GTK_DIALOG(dialog), GTK_RESPONSE_ACCEPT);
1254 gtk_dialog_run(GTK_DIALOG(dialog));
1255 gtk_widget_destroy(dialog);
1257 gdk_threads_leave();
1260 struct client_ops gfio_client_ops = {
1261 .text_op = gfio_text_op,
1262 .disk_util = gfio_disk_util_op,
1263 .thread_status = gfio_thread_status_op,
1264 .group_stats = gfio_group_stats_op,
1265 .eta = gfio_update_eta,
1266 .probe = gfio_probe_op,
1267 .quit = gfio_quit_op,
1268 .add_job = gfio_add_job_op,
1269 .timed_out = gfio_client_timed_out,
1270 .stay_connected = 1,
1273 static void quit_clicked(__attribute__((unused)) GtkWidget *widget,
1274 __attribute__((unused)) gpointer data)
1279 static void *job_thread(void *arg)
1281 fio_handle_clients(&gfio_client_ops);
1285 static int send_job_files(struct gui *ui)
1289 for (i = 0; i < ui->nr_job_files; i++) {
1290 ret = fio_clients_send_ini(ui->job_files[i]);
1294 free(ui->job_files[i]);
1295 ui->job_files[i] = NULL;
1297 while (i < ui->nr_job_files) {
1298 free(ui->job_files[i]);
1299 ui->job_files[i] = NULL;
1306 static void start_job_thread(struct gui *ui)
1308 if (send_job_files(ui)) {
1309 printf("Yeah, I didn't really like those options too much.\n");
1310 gtk_widget_set_sensitive(ui->button[START_JOB_BUTTON], 1);
1315 static void *server_thread(void *arg)
1318 gfio_server_running = 1;
1319 fio_start_server(NULL);
1320 gfio_server_running = 0;
1324 static void gfio_start_server(struct gui *ui)
1326 if (!gfio_server_running) {
1327 gfio_server_running = 1;
1328 pthread_create(&ui->server_t, NULL, server_thread, NULL);
1329 pthread_detach(ui->server_t);
1333 static void start_job_clicked(__attribute__((unused)) GtkWidget *widget,
1336 struct gui *ui = data;
1338 gtk_widget_set_sensitive(ui->button[START_JOB_BUTTON], 0);
1339 start_job_thread(ui);
1342 static void file_open(GtkWidget *w, gpointer data);
1344 static void connect_clicked(GtkWidget *widget, gpointer data)
1346 struct gui *ui = data;
1348 if (!ui->connected) {
1349 if (!ui->nr_job_files)
1350 file_open(widget, data);
1351 gtk_progress_bar_set_text(GTK_PROGRESS_BAR(ui->thread_status_pb), "No jobs running");
1352 gtk_progress_bar_set_fraction(GTK_PROGRESS_BAR(ui->thread_status_pb), 0.0);
1353 if (!fio_clients_connect()) {
1354 pthread_create(&ui->t, NULL, job_thread, NULL);
1355 gtk_widget_set_sensitive(ui->button[CONNECT_BUTTON], 0);
1358 fio_clients_terminate();
1359 gfio_set_connected(ui, 0);
1364 static void add_button(struct gui *ui, int i, GtkWidget *buttonbox,
1365 struct button_spec *buttonspec)
1367 ui->button[i] = gtk_button_new_with_label(buttonspec->buttontext);
1368 g_signal_connect(ui->button[i], "clicked", G_CALLBACK (buttonspec->f), ui);
1369 gtk_box_pack_start(GTK_BOX (ui->buttonbox), ui->button[i], FALSE, FALSE, 3);
1370 gtk_widget_set_tooltip_text(ui->button[i], buttonspeclist[i].tooltiptext);
1371 gtk_widget_set_sensitive(ui->button[i], !buttonspec->start_insensitive);
1374 static void add_buttons(struct gui *ui,
1375 struct button_spec *buttonlist,
1380 for (i = 0; i < nbuttons; i++)
1381 add_button(ui, i, ui->buttonbox, &buttonlist[i]);
1384 static void on_info_bar_response(GtkWidget *widget, gint response,
1387 if (response == GTK_RESPONSE_OK) {
1388 gtk_widget_destroy(widget);
1389 ui.error_info_bar = NULL;
1393 void report_error(GError *error)
1395 if (ui.error_info_bar == NULL) {
1396 ui.error_info_bar = gtk_info_bar_new_with_buttons(GTK_STOCK_OK,
1399 g_signal_connect(ui.error_info_bar, "response", G_CALLBACK(on_info_bar_response), NULL);
1400 gtk_info_bar_set_message_type(GTK_INFO_BAR(ui.error_info_bar),
1403 ui.error_label = gtk_label_new(error->message);
1404 GtkWidget *container = gtk_info_bar_get_content_area(GTK_INFO_BAR(ui.error_info_bar));
1405 gtk_container_add(GTK_CONTAINER(container), ui.error_label);
1407 gtk_box_pack_start(GTK_BOX(ui.vbox), ui.error_info_bar, FALSE, FALSE, 0);
1408 gtk_widget_show_all(ui.vbox);
1411 snprintf(buffer, sizeof(buffer), "Failed to open file.");
1412 gtk_label_set(GTK_LABEL(ui.error_label), buffer);
1416 struct connection_widgets
1423 static void hostname_cb(GtkEntry *entry, gpointer data)
1425 struct connection_widgets *cw = data;
1426 int uses_net = 0, is_localhost = 0;
1431 * Check whether to display the 'auto start backend' box
1432 * or not. Show it if we are a localhost and using network,
1433 * or using a socket.
1435 ctext = gtk_combo_box_get_active_text(GTK_COMBO_BOX(cw->combo));
1436 if (!ctext || !strncmp(ctext, "IPv4", 4) || !strncmp(ctext, "IPv6", 4))
1441 text = gtk_entry_get_text(GTK_ENTRY(cw->hentry));
1442 if (!strcmp(text, "127.0.0.1") || !strcmp(text, "localhost") ||
1443 !strcmp(text, "::1") || !strcmp(text, "ip6-localhost") ||
1444 !strcmp(text, "ip6-loopback"))
1448 if (!uses_net || is_localhost) {
1449 gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(cw->button), 1);
1450 gtk_widget_set_sensitive(cw->button, 1);
1452 gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(cw->button), 0);
1453 gtk_widget_set_sensitive(cw->button, 0);
1457 static int get_connection_details(char **host, int *port, int *type,
1460 GtkWidget *dialog, *box, *vbox, *hbox, *frame, *pentry;
1461 struct connection_widgets cw;
1464 dialog = gtk_dialog_new_with_buttons("Connection details",
1465 GTK_WINDOW(ui.window),
1466 GTK_DIALOG_DESTROY_WITH_PARENT,
1467 GTK_STOCK_OK, GTK_RESPONSE_ACCEPT,
1468 GTK_STOCK_CANCEL, GTK_RESPONSE_REJECT, NULL);
1470 frame = gtk_frame_new("Hostname / socket name");
1471 /* gtk_dialog_get_content_area() is 2.14 and newer */
1472 vbox = GTK_DIALOG(dialog)->vbox;
1473 gtk_box_pack_start(GTK_BOX(vbox), frame, FALSE, FALSE, 5);
1475 box = gtk_vbox_new(FALSE, 6);
1476 gtk_container_add(GTK_CONTAINER(frame), box);
1478 hbox = gtk_hbox_new(TRUE, 10);
1479 gtk_box_pack_start(GTK_BOX(box), hbox, FALSE, FALSE, 0);
1480 cw.hentry = gtk_entry_new();
1481 gtk_entry_set_text(GTK_ENTRY(cw.hentry), "localhost");
1482 gtk_box_pack_start(GTK_BOX(hbox), cw.hentry, TRUE, TRUE, 0);
1484 frame = gtk_frame_new("Port");
1485 gtk_box_pack_start(GTK_BOX(vbox), frame, FALSE, FALSE, 5);
1486 box = gtk_vbox_new(FALSE, 10);
1487 gtk_container_add(GTK_CONTAINER(frame), box);
1489 hbox = gtk_hbox_new(TRUE, 4);
1490 gtk_box_pack_start(GTK_BOX(box), hbox, FALSE, FALSE, 0);
1491 pentry = create_spinbutton(hbox, 1, 65535, FIO_NET_PORT);
1493 frame = gtk_frame_new("Type");
1494 gtk_box_pack_start(GTK_BOX(vbox), frame, FALSE, FALSE, 5);
1495 box = gtk_vbox_new(FALSE, 10);
1496 gtk_container_add(GTK_CONTAINER(frame), box);
1498 hbox = gtk_hbox_new(TRUE, 4);
1499 gtk_box_pack_start(GTK_BOX(box), hbox, FALSE, FALSE, 0);
1501 cw.combo = gtk_combo_box_new_text();
1502 gtk_combo_box_append_text(GTK_COMBO_BOX(cw.combo), "IPv4");
1503 gtk_combo_box_append_text(GTK_COMBO_BOX(cw.combo), "IPv6");
1504 gtk_combo_box_append_text(GTK_COMBO_BOX(cw.combo), "local socket");
1505 gtk_combo_box_set_active(GTK_COMBO_BOX(cw.combo), 0);
1507 gtk_container_add(GTK_CONTAINER(hbox), cw.combo);
1509 frame = gtk_frame_new("Options");
1510 gtk_box_pack_start(GTK_BOX(vbox), frame, FALSE, FALSE, 5);
1511 box = gtk_vbox_new(FALSE, 10);
1512 gtk_container_add(GTK_CONTAINER(frame), box);
1514 hbox = gtk_hbox_new(TRUE, 4);
1515 gtk_box_pack_start(GTK_BOX(box), hbox, FALSE, FALSE, 0);
1517 cw.button = gtk_check_button_new_with_label("Auto-spawn fio backend");
1518 gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(cw.button), 1);
1519 gtk_widget_set_tooltip_text(cw.button, "When running fio locally, it is necessary to have the backend running on the same system. If this is checked, gfio will start the backend automatically for you if it isn't already running.");
1520 gtk_box_pack_start(GTK_BOX(hbox), cw.button, FALSE, FALSE, 6);
1523 * Connect edit signal, so we can show/not-show the auto start button
1525 g_signal_connect(GTK_OBJECT(cw.hentry), "changed", G_CALLBACK(hostname_cb), &cw);
1526 g_signal_connect(GTK_OBJECT(cw.combo), "changed", G_CALLBACK(hostname_cb), &cw);
1528 gtk_widget_show_all(dialog);
1530 if (gtk_dialog_run(GTK_DIALOG(dialog)) != GTK_RESPONSE_ACCEPT) {
1531 gtk_widget_destroy(dialog);
1535 *host = strdup(gtk_entry_get_text(GTK_ENTRY(cw.hentry)));
1536 *port = gtk_spin_button_get_value_as_int(GTK_SPIN_BUTTON(pentry));
1538 typeentry = gtk_combo_box_get_active_text(GTK_COMBO_BOX(cw.combo));
1539 if (!typeentry || !strncmp(typeentry, "IPv4", 4))
1540 *type = Fio_client_ipv4;
1541 else if (!strncmp(typeentry, "IPv6", 4))
1542 *type = Fio_client_ipv6;
1544 *type = Fio_client_socket;
1547 *server_start = gtk_toggle_button_get_active(GTK_TOGGLE_BUTTON(cw.button));
1549 gtk_widget_destroy(dialog);
1553 static void gfio_client_added(struct gui *ui, struct fio_client *client)
1555 struct gfio_client *gc;
1557 gc = malloc(sizeof(*gc));
1558 memset(gc, 0, sizeof(*gc));
1561 client->client_data = gc;
1564 static void file_open(GtkWidget *w, gpointer data)
1567 struct gui *ui = data;
1568 GSList *filenames, *fn_glist;
1569 GtkFileFilter *filter;
1571 int port, type, server_start;
1573 dialog = gtk_file_chooser_dialog_new("Open File",
1574 GTK_WINDOW(ui->window),
1575 GTK_FILE_CHOOSER_ACTION_OPEN,
1576 GTK_STOCK_CANCEL, GTK_RESPONSE_CANCEL,
1577 GTK_STOCK_OPEN, GTK_RESPONSE_ACCEPT,
1579 gtk_file_chooser_set_select_multiple(GTK_FILE_CHOOSER(dialog), TRUE);
1581 filter = gtk_file_filter_new();
1582 gtk_file_filter_add_pattern(filter, "*.fio");
1583 gtk_file_filter_add_pattern(filter, "*.job");
1584 gtk_file_filter_add_pattern(filter, "*.ini");
1585 gtk_file_filter_add_mime_type(filter, "text/fio");
1586 gtk_file_filter_set_name(filter, "Fio job file");
1587 gtk_file_chooser_set_filter(GTK_FILE_CHOOSER(dialog), filter);
1589 if (gtk_dialog_run(GTK_DIALOG(dialog)) != GTK_RESPONSE_ACCEPT) {
1590 gtk_widget_destroy(dialog);
1594 fn_glist = gtk_file_chooser_get_filenames(GTK_FILE_CHOOSER(dialog));
1596 gtk_widget_destroy(dialog);
1598 if (get_connection_details(&host, &port, &type, &server_start))
1601 filenames = fn_glist;
1602 while (filenames != NULL) {
1603 struct fio_client *client;
1605 ui->job_files = realloc(ui->job_files, (ui->nr_job_files + 1) * sizeof(char *));
1606 ui->job_files[ui->nr_job_files] = strdup(filenames->data);
1609 client = fio_client_add_explicit(&gfio_client_ops, host, type, port);
1613 error = g_error_new(g_quark_from_string("fio"), 1,
1614 "Failed to add client %s", host);
1615 report_error(error);
1616 g_error_free(error);
1618 gfio_client_added(ui, client);
1620 g_free(filenames->data);
1621 filenames = g_slist_next(filenames);
1626 gfio_start_server(ui);
1628 g_slist_free(fn_glist);
1631 static void file_save(GtkWidget *w, gpointer data)
1633 struct gui *ui = data;
1636 dialog = gtk_file_chooser_dialog_new("Save File",
1637 GTK_WINDOW(ui->window),
1638 GTK_FILE_CHOOSER_ACTION_SAVE,
1639 GTK_STOCK_CANCEL, GTK_RESPONSE_CANCEL,
1640 GTK_STOCK_SAVE, GTK_RESPONSE_ACCEPT,
1643 gtk_file_chooser_set_do_overwrite_confirmation(GTK_FILE_CHOOSER(dialog), TRUE);
1644 gtk_file_chooser_set_current_name(GTK_FILE_CHOOSER(dialog), "Untitled document");
1646 if (gtk_dialog_run(GTK_DIALOG(dialog)) == GTK_RESPONSE_ACCEPT) {
1649 filename = gtk_file_chooser_get_filename(GTK_FILE_CHOOSER(dialog));
1650 // save_job_file(filename);
1653 gtk_widget_destroy(dialog);
1656 static void view_log_destroy(GtkWidget *w, gpointer data)
1658 struct gui *ui = (struct gui *) data;
1660 gtk_widget_ref(ui->log_tree);
1661 gtk_container_remove(GTK_CONTAINER(w), ui->log_tree);
1662 gtk_widget_destroy(w);
1663 ui->log_view = NULL;
1666 static void view_log(GtkWidget *w, gpointer data)
1668 GtkWidget *win, *scroll, *vbox, *box;
1669 struct gui *ui = (struct gui *) data;
1674 ui->log_view = win = gtk_window_new(GTK_WINDOW_TOPLEVEL);
1675 gtk_window_set_title(GTK_WINDOW(win), "Log");
1676 gtk_window_set_default_size(GTK_WINDOW(win), 700, 500);
1678 scroll = gtk_scrolled_window_new(NULL, NULL);
1680 gtk_container_set_border_width(GTK_CONTAINER(scroll), 5);
1682 gtk_scrolled_window_set_policy(GTK_SCROLLED_WINDOW(scroll), GTK_POLICY_AUTOMATIC, GTK_POLICY_AUTOMATIC);
1684 box = gtk_hbox_new(TRUE, 0);
1685 gtk_box_pack_start_defaults(GTK_BOX(box), ui->log_tree);
1686 g_signal_connect(box, "destroy", G_CALLBACK(view_log_destroy), ui);
1687 gtk_scrolled_window_add_with_viewport(GTK_SCROLLED_WINDOW(scroll), box);
1689 vbox = gtk_vbox_new(TRUE, 5);
1690 gtk_box_pack_start_defaults(GTK_BOX(vbox), scroll);
1692 gtk_container_add(GTK_CONTAINER(win), vbox);
1693 gtk_widget_show_all(win);
1696 static void preferences(GtkWidget *w, gpointer data)
1698 GtkWidget *dialog, *frame, *box, **buttons, *vbox, *font;
1701 dialog = gtk_dialog_new_with_buttons("Preferences",
1702 GTK_WINDOW(ui.window),
1703 GTK_DIALOG_DESTROY_WITH_PARENT,
1704 GTK_STOCK_OK, GTK_RESPONSE_ACCEPT,
1705 GTK_STOCK_CANCEL, GTK_RESPONSE_REJECT,
1708 frame = gtk_frame_new("Debug logging");
1709 gtk_box_pack_start(GTK_BOX(GTK_DIALOG(dialog)->vbox), frame, FALSE, FALSE, 5);
1711 vbox = gtk_vbox_new(FALSE, 6);
1712 gtk_container_add(GTK_CONTAINER(frame), vbox);
1714 box = gtk_hbox_new(FALSE, 6);
1715 gtk_container_add(GTK_CONTAINER(vbox), box);
1717 buttons = malloc(sizeof(GtkWidget *) * FD_DEBUG_MAX);
1719 for (i = 0; i < FD_DEBUG_MAX; i++) {
1721 box = gtk_hbox_new(FALSE, 6);
1722 gtk_container_add(GTK_CONTAINER(vbox), box);
1726 buttons[i] = gtk_check_button_new_with_label(debug_levels[i].name);
1727 gtk_widget_set_tooltip_text(buttons[i], debug_levels[i].help);
1728 gtk_box_pack_start(GTK_BOX(box), buttons[i], FALSE, FALSE, 6);
1731 frame = gtk_frame_new("Graph font");
1732 gtk_box_pack_start(GTK_BOX(GTK_DIALOG(dialog)->vbox), frame, FALSE, FALSE, 5);
1733 vbox = gtk_vbox_new(FALSE, 6);
1734 gtk_container_add(GTK_CONTAINER(frame), vbox);
1736 font = gtk_font_button_new();
1737 gtk_box_pack_start(GTK_BOX(vbox), font, FALSE, FALSE, 5);
1739 gtk_widget_show_all(dialog);
1741 if (gtk_dialog_run(GTK_DIALOG(dialog)) != GTK_RESPONSE_ACCEPT) {
1742 gtk_widget_destroy(dialog);
1746 for (i = 0; i < FD_DEBUG_MAX; i++) {
1749 set = gtk_toggle_button_get_active(GTK_TOGGLE_BUTTON(buttons[i]));
1751 fio_debug |= (1UL << i);
1754 gfio_graph_font = strdup(gtk_font_button_get_font_name(GTK_FONT_BUTTON(font)));
1755 gtk_widget_destroy(dialog);
1758 static void about_dialog(GtkWidget *w, gpointer data)
1760 const char *authors[] = {
1761 "Jens Axboe <axboe@kernel.dk>",
1762 "Stephen Carmeron <stephenmcameron@gmail.com>",
1765 const char *license[] = {
1766 "Fio is free software; you can redistribute it and/or modify "
1767 "it under the terms of the GNU General Public License as published by "
1768 "the Free Software Foundation; either version 2 of the License, or "
1769 "(at your option) any later version.\n",
1770 "Fio is distributed in the hope that it will be useful, "
1771 "but WITHOUT ANY WARRANTY; without even the implied warranty of "
1772 "MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the "
1773 "GNU General Public License for more details.\n",
1774 "You should have received a copy of the GNU General Public License "
1775 "along with Fio; if not, write to the Free Software Foundation, Inc., "
1776 "51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA\n"
1778 char *license_trans;
1780 license_trans = g_strconcat(license[0], "\n", license[1], "\n",
1781 license[2], "\n", NULL);
1783 gtk_show_about_dialog(NULL,
1784 "program-name", "gfio",
1785 "comments", "Gtk2 UI for fio",
1786 "license", license_trans,
1787 "website", "http://git.kernel.dk/?p=fio.git;a=summary",
1789 "version", fio_version_string,
1790 "copyright", "© 2012 Jens Axboe <axboe@kernel.dk>",
1791 "logo-icon-name", "fio",
1793 "wrap-license", TRUE,
1796 g_free (license_trans);
1799 static GtkActionEntry menu_items[] = {
1800 { "FileMenuAction", GTK_STOCK_FILE, "File", NULL, NULL, NULL},
1801 { "ViewMenuAction", GTK_STOCK_FILE, "View", NULL, NULL, NULL},
1802 { "HelpMenuAction", GTK_STOCK_HELP, "Help", NULL, NULL, NULL},
1803 { "OpenFile", GTK_STOCK_OPEN, NULL, "<Control>O", NULL, G_CALLBACK(file_open) },
1804 { "SaveFile", GTK_STOCK_SAVE, NULL, "<Control>S", NULL, G_CALLBACK(file_save) },
1805 { "Preferences", GTK_STOCK_PREFERENCES, NULL, "<Control>p", NULL, G_CALLBACK(preferences) },
1806 { "ViewLog", NULL, "Log", "<Control>l", NULL, G_CALLBACK(view_log) },
1807 { "Quit", GTK_STOCK_QUIT, NULL, "<Control>Q", NULL, G_CALLBACK(quit_clicked) },
1808 { "About", GTK_STOCK_ABOUT, NULL, NULL, NULL, G_CALLBACK(about_dialog) },
1810 static gint nmenu_items = sizeof(menu_items) / sizeof(menu_items[0]);
1812 static const gchar *ui_string = " \
1814 <menubar name=\"MainMenu\"> \
1815 <menu name=\"FileMenu\" action=\"FileMenuAction\"> \
1816 <menuitem name=\"Open\" action=\"OpenFile\" /> \
1817 <menuitem name=\"Save\" action=\"SaveFile\" /> \
1818 <separator name=\"Separator\"/> \
1819 <menuitem name=\"Preferences\" action=\"Preferences\" /> \
1820 <separator name=\"Separator2\"/> \
1821 <menuitem name=\"Quit\" action=\"Quit\" /> \
1823 <menu name=\"ViewMenu\" action=\"ViewMenuAction\"> \
1824 <menuitem name=\"Log\" action=\"ViewLog\" /> \
1826 <menu name=\"Help\" action=\"HelpMenuAction\"> \
1827 <menuitem name=\"About\" action=\"About\" /> \
1833 static GtkWidget *get_menubar_menu(GtkWidget *window, GtkUIManager *ui_manager,
1836 GtkActionGroup *action_group = gtk_action_group_new("Menu");
1839 action_group = gtk_action_group_new("Menu");
1840 gtk_action_group_add_actions(action_group, menu_items, nmenu_items, ui);
1842 gtk_ui_manager_insert_action_group(ui_manager, action_group, 0);
1843 gtk_ui_manager_add_ui_from_string(GTK_UI_MANAGER(ui_manager), ui_string, -1, &error);
1845 gtk_window_add_accel_group(GTK_WINDOW(window), gtk_ui_manager_get_accel_group(ui_manager));
1846 return gtk_ui_manager_get_widget(ui_manager, "/MainMenu");
1849 void gfio_ui_setup(GtkSettings *settings, GtkWidget *menubar,
1850 GtkWidget *vbox, GtkUIManager *ui_manager)
1852 gtk_box_pack_start(GTK_BOX(vbox), menubar, FALSE, FALSE, 0);
1855 static void init_ui(int *argc, char **argv[], struct gui *ui)
1857 GtkSettings *settings;
1858 GtkUIManager *uimanager;
1859 GtkWidget *menu, *probe, *probe_frame, *probe_box;
1862 memset(ui, 0, sizeof(*ui));
1864 /* Magical g*thread incantation, you just need this thread stuff.
1865 * Without it, the update that happens in gfio_update_thread_status
1866 * doesn't really happen in a timely fashion, you need expose events
1868 if (!g_thread_supported())
1869 g_thread_init(NULL);
1872 gtk_init(argc, argv);
1873 settings = gtk_settings_get_default();
1874 gtk_settings_set_long_property(settings, "gtk_tooltip_timeout", 10, "gfio setting");
1877 ui->window = gtk_window_new(GTK_WINDOW_TOPLEVEL);
1878 gtk_window_set_title(GTK_WINDOW(ui->window), "fio");
1879 gtk_window_set_default_size(GTK_WINDOW(ui->window), 700, 700);
1881 g_signal_connect(ui->window, "delete-event", G_CALLBACK(quit_clicked), NULL);
1882 g_signal_connect(ui->window, "destroy", G_CALLBACK(quit_clicked), NULL);
1884 ui->vbox = gtk_vbox_new(FALSE, 0);
1885 gtk_container_add(GTK_CONTAINER (ui->window), ui->vbox);
1887 uimanager = gtk_ui_manager_new();
1888 menu = get_menubar_menu(ui->window, uimanager, ui);
1889 gfio_ui_setup(settings, menu, ui->vbox, uimanager);
1892 * Set up alignments for widgets at the top of ui,
1893 * align top left, expand horizontally but not vertically
1895 ui->topalign = gtk_alignment_new(0, 0, 1, 0);
1896 ui->topvbox = gtk_vbox_new(FALSE, 3);
1897 gtk_container_add(GTK_CONTAINER(ui->topalign), ui->topvbox);
1898 gtk_box_pack_start(GTK_BOX(ui->vbox), ui->topalign, FALSE, FALSE, 0);
1900 probe = gtk_frame_new("Job");
1901 gtk_box_pack_start(GTK_BOX(ui->topvbox), probe, TRUE, FALSE, 3);
1902 probe_frame = gtk_vbox_new(FALSE, 3);
1903 gtk_container_add(GTK_CONTAINER(probe), probe_frame);
1905 probe_box = gtk_hbox_new(FALSE, 3);
1906 gtk_box_pack_start(GTK_BOX(probe_frame), probe_box, TRUE, FALSE, 3);
1907 ui->probe.hostname = new_info_label_in_frame(probe_box, "Host");
1908 ui->probe.os = new_info_label_in_frame(probe_box, "OS");
1909 ui->probe.arch = new_info_label_in_frame(probe_box, "Architecture");
1910 ui->probe.fio_ver = new_info_label_in_frame(probe_box, "Fio version");
1912 probe_box = gtk_hbox_new(FALSE, 3);
1913 gtk_box_pack_start(GTK_BOX(probe_frame), probe_box, TRUE, FALSE, 3);
1915 ui->eta.name = new_info_entry_in_frame(probe_box, "Name");
1916 ui->eta.iotype = new_info_entry_in_frame(probe_box, "IO");
1917 ui->eta.ioengine = new_info_entry_in_frame(probe_box, "IO Engine");
1918 ui->eta.iodepth = new_info_entry_in_frame(probe_box, "IO Depth");
1919 ui->eta.jobs = new_info_entry_in_frame(probe_box, "Jobs");
1920 ui->eta.files = new_info_entry_in_frame(probe_box, "Open files");
1922 probe_box = gtk_hbox_new(FALSE, 3);
1923 gtk_box_pack_start(GTK_BOX(probe_frame), probe_box, TRUE, FALSE, 3);
1924 ui->eta.read_bw = new_info_entry_in_frame(probe_box, "Read BW");
1925 ui->eta.read_iops = new_info_entry_in_frame(probe_box, "IOPS");
1926 ui->eta.write_bw = new_info_entry_in_frame(probe_box, "Write BW");
1927 ui->eta.write_iops = new_info_entry_in_frame(probe_box, "IOPS");
1930 * Only add this if we have a commit rate
1933 probe_box = gtk_hbox_new(FALSE, 3);
1934 gtk_box_pack_start(GTK_BOX(probe_frame), probe_box, TRUE, FALSE, 3);
1936 ui->eta.cr_bw = new_info_label_in_frame(probe_box, "Commit BW");
1937 ui->eta.cr_iops = new_info_label_in_frame(probe_box, "Commit IOPS");
1939 ui->eta.cw_bw = new_info_label_in_frame(probe_box, "Commit BW");
1940 ui->eta.cw_iops = new_info_label_in_frame(probe_box, "Commit IOPS");
1944 * Set up a drawing area and IOPS and bandwidth graphs
1946 gdk_color_parse("white", &white);
1947 ui->drawing_area = gtk_drawing_area_new();
1948 ui->drawing_area_xdim = DRAWING_AREA_XDIM;
1949 ui->drawing_area_ydim = DRAWING_AREA_YDIM;
1950 gtk_widget_set_size_request(GTK_WIDGET(ui->drawing_area),
1951 ui->drawing_area_xdim, ui->drawing_area_ydim);
1952 gtk_widget_modify_bg(ui->drawing_area, GTK_STATE_NORMAL, &white);
1953 g_signal_connect(G_OBJECT(ui->drawing_area), "expose_event",
1954 G_CALLBACK (on_expose_drawing_area), ui);
1955 g_signal_connect(G_OBJECT(ui->drawing_area), "configure_event",
1956 G_CALLBACK (on_config_drawing_area), ui);
1957 ui->scrolled_window = gtk_scrolled_window_new(NULL, NULL);
1958 gtk_scrolled_window_set_policy(GTK_SCROLLED_WINDOW(ui->scrolled_window),
1959 GTK_POLICY_AUTOMATIC, GTK_POLICY_AUTOMATIC);
1960 gtk_scrolled_window_add_with_viewport(GTK_SCROLLED_WINDOW(ui->scrolled_window),
1962 gtk_box_pack_start(GTK_BOX(ui->vbox), ui->scrolled_window,
1965 setup_iops_graph(ui);
1966 setup_bandwidth_graph(ui);
1969 * Set up alignments for widgets at the bottom of ui,
1970 * align bottom left, expand horizontally but not vertically
1972 ui->bottomalign = gtk_alignment_new(0, 1, 1, 0);
1973 ui->buttonbox = gtk_hbox_new(FALSE, 0);
1974 gtk_container_add(GTK_CONTAINER(ui->bottomalign), ui->buttonbox);
1975 gtk_box_pack_start(GTK_BOX(ui->vbox), ui->bottomalign,
1978 add_buttons(ui, buttonspeclist, ARRAYSIZE(buttonspeclist));
1981 * Set up thread status progress bar
1983 ui->thread_status_pb = gtk_progress_bar_new();
1984 gtk_progress_bar_set_fraction(GTK_PROGRESS_BAR(ui->thread_status_pb), 0.0);
1985 gtk_progress_bar_set_text(GTK_PROGRESS_BAR(ui->thread_status_pb), "No connections");
1986 gtk_container_add(GTK_CONTAINER(ui->buttonbox), ui->thread_status_pb);
1988 gfio_ui_setup_log(ui);
1990 gtk_widget_show_all(ui->window);
1993 int main(int argc, char *argv[], char *envp[])
1995 if (initialize_fio(envp))
1997 if (fio_init_options())
2000 init_ui(&argc, &argv, &ui);
2002 gdk_threads_enter();
2004 gdk_threads_leave();