blob: 3de07c9e25bb98d78e8de85cd958c6335f5c6a08 [file] [log] [blame]
Stephen M. Cameronff1f3282012-02-24 08:17:30 +01001/*
2 * gfio - gui front end for fio - the flexible io tester
3 *
4 * Copyright (C) 2012 Stephen M. Cameron <stephenmcameron@gmail.com>
Jens Axboec0187f32012-03-06 15:39:15 +01005 * Copyright (C) 2012 Jens Axboe <axboe@kernel.dk>
Stephen M. Cameronff1f3282012-02-24 08:17:30 +01006 *
7 * The license below covers all files distributed with fio unless otherwise
8 * noted in the file itself.
9 *
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.
13 *
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.
18 *
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
22 *
23 */
Stephen M. Cameron8232e282012-02-24 08:17:31 +010024#include <locale.h>
Stephen M. Cameron60f6b332012-02-24 08:17:32 +010025#include <malloc.h>
Stephen M. Cameron8232e282012-02-24 08:17:31 +010026
Stephen M. Cameron5b7573a2012-02-24 08:17:31 +010027#include <glib.h>
Jens Axboe2fd3bb02012-03-07 08:07:39 +010028#include <cairo.h>
Stephen M. Cameronff1f3282012-02-24 08:17:30 +010029#include <gtk/gtk.h>
30
Stephen M. Cameron8232e282012-02-24 08:17:31 +010031#include "fio.h"
Jens Axboe2fd3bb02012-03-07 08:07:39 +010032#include "graph.h"
Stephen M. Cameron8232e282012-02-24 08:17:31 +010033
Jens Axboe63a130b2012-03-06 20:08:59 +010034static int gfio_server_running;
Jens Axboef3e84402012-03-07 13:14:32 +010035static const char *gfio_graph_font;
Jens Axboe63a130b2012-03-06 20:08:59 +010036
Jens Axboe3e47bd22012-02-29 13:45:02 +010037static void gfio_update_thread_status(char *status_message, double perc);
38
Stephen M. Cameronf3074002012-02-24 08:17:30 +010039#define ARRAYSIZE(x) (sizeof((x)) / (sizeof((x)[0])))
40
41typedef void (*clickfunction)(GtkWidget *widget, gpointer data);
42
Jens Axboe3e47bd22012-02-29 13:45:02 +010043static void connect_clicked(GtkWidget *widget, gpointer data);
Stephen M. Cameronf3074002012-02-24 08:17:30 +010044static void start_job_clicked(GtkWidget *widget, gpointer data);
45
46static struct button_spec {
47 const char *buttontext;
48 clickfunction f;
49 const char *tooltiptext;
Jens Axboe3e47bd22012-02-29 13:45:02 +010050 const int start_insensitive;
Stephen M. Cameronf3074002012-02-24 08:17:30 +010051} buttonspeclist[] = {
Jens Axboe3e47bd22012-02-29 13:45:02 +010052#define CONNECT_BUTTON 0
53#define START_JOB_BUTTON 1
54 { "Connect", connect_clicked, "Connect to host", 0 },
Stephen M. Cameronf3074002012-02-24 08:17:30 +010055 { "Start Job",
56 start_job_clicked,
Jens Axboe3e47bd22012-02-29 13:45:02 +010057 "Send current fio job to fio server to be executed", 1 },
Stephen M. Cameronf3074002012-02-24 08:17:30 +010058};
59
Jens Axboe843ad232012-02-29 11:44:53 +010060struct probe_widget {
61 GtkWidget *hostname;
62 GtkWidget *os;
63 GtkWidget *arch;
64 GtkWidget *fio_ver;
65};
66
Jens Axboe3e47bd22012-02-29 13:45:02 +010067struct eta_widget {
Jens Axboe807f9972012-03-02 10:25:24 +010068 GtkWidget *name;
69 GtkWidget *iotype;
70 GtkWidget *ioengine;
71 GtkWidget *iodepth;
Jens Axboe3e47bd22012-02-29 13:45:02 +010072 GtkWidget *jobs;
73 GtkWidget *files;
74 GtkWidget *read_bw;
75 GtkWidget *read_iops;
76 GtkWidget *cr_bw;
77 GtkWidget *cr_iops;
78 GtkWidget *write_bw;
79 GtkWidget *write_iops;
80 GtkWidget *cw_bw;
81 GtkWidget *cw_iops;
82};
83
Stephen M. Cameronff1f3282012-02-24 08:17:30 +010084struct gui {
85 GtkWidget *window;
Stephen M. Cameron5b7573a2012-02-24 08:17:31 +010086 GtkWidget *vbox;
Stephen M. Cameronc36f98d2012-02-24 08:17:32 +010087 GtkWidget *topvbox;
88 GtkWidget *topalign;
89 GtkWidget *bottomalign;
Stephen M. Cameron04cc6b72012-02-24 08:17:31 +010090 GtkWidget *thread_status_pb;
Stephen M. Cameronf3074002012-02-24 08:17:30 +010091 GtkWidget *buttonbox;
92 GtkWidget *button[ARRAYSIZE(buttonspeclist)];
Stephen M. Cameron736f2df2012-02-24 08:17:32 +010093 GtkWidget *scrolled_window;
Jens Axboe2fd3bb02012-03-07 08:07:39 +010094#define DRAWING_AREA_XDIM 1000
95#define DRAWING_AREA_YDIM 400
96 GtkWidget *drawing_area;
Stephen M. Cameron3ea48b82012-03-07 19:40:58 +010097 int drawing_area_xdim;
98 int drawing_area_ydim;
Jens Axboe0420ba62012-02-29 11:16:52 +010099 GtkWidget *error_info_bar;
100 GtkWidget *error_label;
Jens Axboef9d40b42012-03-06 09:52:49 +0100101 GtkWidget *results_notebook;
102 GtkWidget *results_window;
Jens Axboe9b260bd2012-03-06 11:02:52 +0100103 GtkListStore *log_model;
104 GtkWidget *log_tree;
Jens Axboe4cbe7212012-03-06 13:36:17 +0100105 GtkWidget *log_view;
Stephen M. Cameron736f2df2012-02-24 08:17:32 +0100106 GtkTextBuffer *text;
Jens Axboe843ad232012-02-29 11:44:53 +0100107 struct probe_widget probe;
Jens Axboe3e47bd22012-02-29 13:45:02 +0100108 struct eta_widget eta;
Jens Axboe3ec62ec2012-03-01 12:01:29 +0100109 int connected;
Stephen M. Cameron25927252012-02-24 08:17:31 +0100110 pthread_t t;
Jens Axboe63a130b2012-03-06 20:08:59 +0100111 pthread_t server_t;
Jens Axboe0420ba62012-02-29 11:16:52 +0100112
Jens Axboe2fd3bb02012-03-07 08:07:39 +0100113 struct graph *iops_graph;
114 struct graph *bandwidth_graph;
Jens Axboe3ec62ec2012-03-01 12:01:29 +0100115 struct fio_client *client;
Jens Axboe0420ba62012-02-29 11:16:52 +0100116 int nr_job_files;
117 char **job_files;
Stephen M. Cameron5b7573a2012-02-24 08:17:31 +0100118} ui;
Stephen M. Cameronff1f3282012-02-24 08:17:30 +0100119
Jens Axboee0681f32012-03-06 12:14:42 +0100120struct gfio_client {
121 struct gui *ui;
122 GtkWidget *results_widget;
123 GtkWidget *disk_util_frame;
124};
125
Jens Axboe2fd3bb02012-03-07 08:07:39 +0100126static void setup_iops_graph(struct gui *ui)
127{
128 if (ui->iops_graph)
129 graph_free(ui->iops_graph);
Jens Axboe87d5f272012-03-07 12:31:40 +0100130 ui->iops_graph = graph_new(DRAWING_AREA_XDIM / 2.0,
Jens Axboef3e84402012-03-07 13:14:32 +0100131 DRAWING_AREA_YDIM, gfio_graph_font);
Jens Axboe2fd3bb02012-03-07 08:07:39 +0100132 graph_title(ui->iops_graph, "IOPS");
Stephen M. Cameronb04ad8d2012-03-07 14:49:37 +0100133 graph_x_title(ui->iops_graph, "Time (secs)");
Jens Axboe2fd3bb02012-03-07 08:07:39 +0100134 graph_add_label(ui->iops_graph, "Read IOPS");
135 graph_add_label(ui->iops_graph, "Write IOPS");
Jens Axboe9f4883a2012-03-07 16:22:50 +0100136 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);
Stephen M. Cameronfe8afdd2012-03-07 19:34:19 +0100138 line_graph_set_data_count_limit(ui->iops_graph, 100);
Jens Axboe2fd3bb02012-03-07 08:07:39 +0100139}
140
141static void setup_bandwidth_graph(struct gui *ui)
142{
143 if (ui->bandwidth_graph)
144 graph_free(ui->bandwidth_graph);
Jens Axboe87d5f272012-03-07 12:31:40 +0100145 ui->bandwidth_graph = graph_new(DRAWING_AREA_XDIM / 2.0,
Jens Axboef3e84402012-03-07 13:14:32 +0100146 DRAWING_AREA_YDIM, gfio_graph_font);
Jens Axboe2fd3bb02012-03-07 08:07:39 +0100147 graph_title(ui->bandwidth_graph, "Bandwidth");
Stephen M. Cameronb04ad8d2012-03-07 14:49:37 +0100148 graph_x_title(ui->bandwidth_graph, "Time (secs)");
Jens Axboe2fd3bb02012-03-07 08:07:39 +0100149 graph_add_label(ui->bandwidth_graph, "Read Bandwidth");
150 graph_add_label(ui->bandwidth_graph, "Write Bandwidth");
Jens Axboe9f4883a2012-03-07 16:22:50 +0100151 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);
Stephen M. Cameronfe8afdd2012-03-07 19:34:19 +0100153 line_graph_set_data_count_limit(ui->bandwidth_graph, 100);
Jens Axboe2fd3bb02012-03-07 08:07:39 +0100154}
155
Jens Axboe8663ea62012-03-02 14:04:30 +0100156static void clear_ui_info(struct gui *ui)
157{
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), "");
Jens Axboeca850992012-03-05 20:04:43 +0100162 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), "");
Jens Axboe8663ea62012-03-02 14:04:30 +0100172}
173
Jens Axboe3650a3c2012-03-05 14:09:03 +0100174static GtkWidget *new_info_entry_in_frame(GtkWidget *box, const char *label)
175{
176 GtkWidget *entry, *frame;
177
178 frame = gtk_frame_new(label);
179 entry = gtk_entry_new();
Jens Axboe1c1e4a52012-03-05 14:37:38 +0100180 gtk_entry_set_editable(GTK_ENTRY(entry), 0);
Jens Axboe3650a3c2012-03-05 14:09:03 +0100181 gtk_box_pack_start(GTK_BOX(box), frame, TRUE, TRUE, 3);
182 gtk_container_add(GTK_CONTAINER(frame), entry);
183
184 return entry;
185}
186
187static GtkWidget *new_info_label_in_frame(GtkWidget *box, const char *label)
188{
189 GtkWidget *label_widget;
190 GtkWidget *frame;
191
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);
196
197 return label_widget;
198}
199
200static GtkWidget *create_spinbutton(GtkWidget *hbox, double min, double max, double defval)
201{
202 GtkWidget *button, *box;
203
204 box = gtk_hbox_new(FALSE, 3);
205 gtk_container_add(GTK_CONTAINER(hbox), box);
206
207 button = gtk_spin_button_new_with_range(min, max, 1.0);
208 gtk_box_pack_start(GTK_BOX(box), button, TRUE, TRUE, 0);
209
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);
212
213 return button;
214}
215
Jens Axboe3ec62ec2012-03-01 12:01:29 +0100216static void gfio_set_connected(struct gui *ui, int connected)
217{
218 if (connected) {
219 gtk_widget_set_sensitive(ui->button[START_JOB_BUTTON], 1);
220 ui->connected = 1;
221 gtk_button_set_label(GTK_BUTTON(ui->button[CONNECT_BUTTON]), "Disconnect");
Jens Axboe88f6e7a2012-03-06 12:55:29 +0100222 gtk_widget_set_sensitive(ui->button[CONNECT_BUTTON], 1);
Jens Axboe3ec62ec2012-03-01 12:01:29 +0100223 } else {
224 ui->connected = 0;
225 gtk_button_set_label(GTK_BUTTON(ui->button[CONNECT_BUTTON]), "Connect");
226 gtk_widget_set_sensitive(ui->button[START_JOB_BUTTON], 0);
Jens Axboe63a130b2012-03-06 20:08:59 +0100227 gtk_widget_set_sensitive(ui->button[CONNECT_BUTTON], 1);
Jens Axboe3ec62ec2012-03-01 12:01:29 +0100228 }
229}
230
Jens Axboe3650a3c2012-03-05 14:09:03 +0100231static void label_set_int_value(GtkWidget *entry, unsigned int val)
232{
233 char tmp[80];
234
235 sprintf(tmp, "%u", val);
236 gtk_label_set_text(GTK_LABEL(entry), tmp);
237}
238
239static void entry_set_int_value(GtkWidget *entry, unsigned int val)
240{
241 char tmp[80];
242
243 sprintf(tmp, "%u", val);
244 gtk_entry_set_text(GTK_ENTRY(entry), tmp);
245}
246
Jens Axboea2697902012-03-05 16:43:49 +0100247#define ALIGN_LEFT 1
248#define ALIGN_RIGHT 2
249#define INVISIBLE 4
250#define UNSORTABLE 8
251
252GtkTreeViewColumn *tree_view_column(GtkWidget *tree_view, int index, const char *title, unsigned int flags)
253{
254 GtkCellRenderer *renderer;
255 GtkTreeViewColumn *col;
256 double xalign = 0.0; /* left as default */
257 PangoAlignment align;
258 gboolean visible;
259
260 align = (flags & ALIGN_LEFT) ? PANGO_ALIGN_LEFT :
261 (flags & ALIGN_RIGHT) ? PANGO_ALIGN_RIGHT :
262 PANGO_ALIGN_CENTER;
263 visible = !(flags & INVISIBLE);
264
265 renderer = gtk_cell_renderer_text_new();
266 col = gtk_tree_view_column_new();
267
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);
275 switch (align) {
276 case PANGO_ALIGN_LEFT:
277 xalign = 0.0;
278 break;
279 case PANGO_ALIGN_CENTER:
280 xalign = 0.5;
281 break;
282 case PANGO_ALIGN_RIGHT:
283 xalign = 1.0;
284 break;
285 }
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);
289 return col;
290}
291
Jens Axboe9b260bd2012-03-06 11:02:52 +0100292static void gfio_ui_setup_log(struct gui *ui)
293{
294 GtkTreeSelection *selection;
295 GtkListStore *model;
296 GtkWidget *tree_view;
297
298 model = gtk_list_store_new(4, G_TYPE_STRING, G_TYPE_STRING, G_TYPE_INT, G_TYPE_STRING);
299
300 tree_view = gtk_tree_view_new_with_model(GTK_TREE_MODEL(model));
301 gtk_widget_set_can_focus(tree_view, FALSE);
302
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);
Jens Axboe661f7412012-03-06 13:55:45 +0100305 g_object_set(G_OBJECT(tree_view), "headers-visible", TRUE,
306 "enable-grid-lines", GTK_TREE_VIEW_GRID_LINES_BOTH, NULL);
Jens Axboe9b260bd2012-03-06 11:02:52 +0100307
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);
Jens Axboef095d562012-03-06 13:49:12 +0100311 tree_view_column(tree_view, 3, "Text", ALIGN_LEFT | UNSORTABLE);
Jens Axboe9b260bd2012-03-06 11:02:52 +0100312
313 ui->log_model = model;
314 ui->log_tree = tree_view;
315}
316
Jens Axboea2697902012-03-05 16:43:49 +0100317static GtkWidget *gfio_output_clat_percentiles(unsigned int *ovals,
318 fio_fp64_t *plist,
319 unsigned int len,
320 const char *base,
321 unsigned int scale)
322{
323 GType types[FIO_IO_U_LIST_MAX_LEN];
324 GtkWidget *tree_view;
325 GtkTreeSelection *selection;
326 GtkListStore *model;
327 GtkTreeIter iter;
328 int i;
329
330 for (i = 0; i < len; i++)
331 types[i] = G_TYPE_INT;
332
333 model = gtk_list_store_newv(len, types);
334
335 tree_view = gtk_tree_view_new_with_model(GTK_TREE_MODEL(model));
336 gtk_widget_set_can_focus(tree_view, FALSE);
337
Jens Axboe661f7412012-03-06 13:55:45 +0100338 g_object_set(G_OBJECT(tree_view), "headers-visible", TRUE,
339 "enable-grid-lines", GTK_TREE_VIEW_GRID_LINES_BOTH, NULL);
340
Jens Axboea2697902012-03-05 16:43:49 +0100341 selection = gtk_tree_view_get_selection(GTK_TREE_VIEW(tree_view));
342 gtk_tree_selection_set_mode(GTK_TREE_SELECTION(selection), GTK_SELECTION_BROWSE);
343
344 for (i = 0; i < len; i++) {
345 char fbuf[8];
346
347 sprintf(fbuf, "%2.2f%%", plist[i].u.f);
348 tree_view_column(tree_view, i, fbuf, ALIGN_RIGHT | UNSORTABLE);
349 }
350
351 gtk_list_store_append(model, &iter);
352
Jens Axboee0681f32012-03-06 12:14:42 +0100353 for (i = 0; i < len; i++) {
354 if (scale)
355 ovals[i] = (ovals[i] + 999) / 1000;
Jens Axboea2697902012-03-05 16:43:49 +0100356 gtk_list_store_set(model, &iter, i, ovals[i], -1);
Jens Axboee0681f32012-03-06 12:14:42 +0100357 }
Jens Axboea2697902012-03-05 16:43:49 +0100358
359 return tree_view;
360}
361
362static void gfio_show_clat_percentiles(GtkWidget *vbox, struct thread_stat *ts,
363 int ddir)
364{
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;
369 const char *base;
370 GtkWidget *tree_view, *frame, *hbox;
371 char tmp[64];
372
373 len = calc_clat_percentiles(io_u_plat, nr, plist, &ovals, &maxv, &minv);
374 if (!len)
375 goto out;
376
377 /*
378 * We default to usecs, but if the value range is such that we
379 * should scale down to msecs, do that.
380 */
381 if (minv > 2000 && maxv > 99999) {
382 scale_down = 1;
383 base = "msec";
384 } else {
385 scale_down = 0;
386 base = "usec";
387 }
388
389 tree_view = gfio_output_clat_percentiles(ovals, plist, len, base, scale_down);
390
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);
394
395 hbox = gtk_hbox_new(FALSE, 3);
396 gtk_container_add(GTK_CONTAINER(frame), hbox);
397
398 gtk_box_pack_start(GTK_BOX(hbox), tree_view, TRUE, FALSE, 3);
399out:
400 if (ovals)
401 free(ovals);
402}
403
Jens Axboe3650a3c2012-03-05 14:09:03 +0100404static void gfio_show_lat(GtkWidget *vbox, const char *name, unsigned long min,
405 unsigned long max, double mean, double dev)
406{
407 const char *base = "(usec)";
Jens Axboe1c1e4a52012-03-05 14:37:38 +0100408 GtkWidget *hbox, *label, *frame;
Jens Axboe3650a3c2012-03-05 14:09:03 +0100409 char *minp, *maxp;
410 char tmp[64];
411
412 if (!usec_to_msec(&min, &max, &mean, &dev))
413 base = "(msec)";
414
415 minp = num2str(min, 6, 1, 0);
416 maxp = num2str(max, 6, 1, 0);
417
Jens Axboe3650a3c2012-03-05 14:09:03 +0100418 sprintf(tmp, "%s %s", name, base);
419 frame = gtk_frame_new(tmp);
420 gtk_box_pack_start(GTK_BOX(vbox), frame, FALSE, FALSE, 5);
421
Jens Axboe3650a3c2012-03-05 14:09:03 +0100422 hbox = gtk_hbox_new(FALSE, 3);
Jens Axboe1c1e4a52012-03-05 14:37:38 +0100423 gtk_container_add(GTK_CONTAINER(frame), hbox);
Jens Axboe3650a3c2012-03-05 14:09:03 +0100424
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);
435
436 free(minp);
437 free(maxp);
438
439}
440
Jens Axboeca850992012-03-05 20:04:43 +0100441#define GFIO_CLAT 1
442#define GFIO_SLAT 2
443#define GFIO_LAT 4
444
Jens Axboe3650a3c2012-03-05 14:09:03 +0100445static void gfio_show_ddir_status(GtkWidget *mbox, struct group_run_stats *rs,
446 struct thread_stat *ts, int ddir)
447{
448 const char *ddir_label[2] = { "Read", "Write" };
Jens Axboe0b761302012-03-05 20:44:11 +0100449 GtkWidget *frame, *label, *box, *vbox, *main_vbox;
Jens Axboee0681f32012-03-06 12:14:42 +0100450 unsigned long min[3], max[3], runt;
Jens Axboe3650a3c2012-03-05 14:09:03 +0100451 unsigned long long bw, iops;
Jens Axboeca850992012-03-05 20:04:43 +0100452 unsigned int flags = 0;
Jens Axboee0681f32012-03-06 12:14:42 +0100453 double mean[3], dev[3];
Jens Axboe3650a3c2012-03-05 14:09:03 +0100454 char *io_p, *bw_p, *iops_p;
455 int i2p;
456
457 if (!ts->runtime[ddir])
458 return;
459
460 i2p = is_power_of_2(rs->kb_base);
461 runt = ts->runtime[ddir];
462
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);
466
467 iops = (1000 * (uint64_t)ts->total_io_u[ddir]) / runt;
468 iops_p = num2str(iops, 6, 1, 0);
469
470 box = gtk_hbox_new(FALSE, 3);
471 gtk_box_pack_start(GTK_BOX(mbox), box, TRUE, FALSE, 3);
472
473 frame = gtk_frame_new(ddir_label[ddir]);
474 gtk_box_pack_start(GTK_BOX(box), frame, FALSE, FALSE, 5);
475
Jens Axboe0b761302012-03-05 20:44:11 +0100476 main_vbox = gtk_vbox_new(FALSE, 3);
477 gtk_container_add(GTK_CONTAINER(frame), main_vbox);
Jens Axboe3650a3c2012-03-05 14:09:03 +0100478
479 box = gtk_hbox_new(FALSE, 3);
Jens Axboe0b761302012-03-05 20:44:11 +0100480 gtk_box_pack_start(GTK_BOX(main_vbox), box, TRUE, FALSE, 3);
Jens Axboe3650a3c2012-03-05 14:09:03 +0100481
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]);
490
Jens Axboee0681f32012-03-06 12:14:42 +0100491 if (calc_lat(&ts->bw_stat[ddir], &min[0], &max[0], &mean[0], &dev[0])) {
Jens Axboeca850992012-03-05 20:04:43 +0100492 double p_of_agg = 100.0;
493 const char *bw_str = "KB";
494 char tmp[32];
495
496 if (rs->agg[ddir]) {
Jens Axboee0681f32012-03-06 12:14:42 +0100497 p_of_agg = mean[0] * 100 / (double) rs->agg[ddir];
Jens Axboeca850992012-03-05 20:04:43 +0100498 if (p_of_agg > 100.0)
499 p_of_agg = 100.0;
500 }
501
Jens Axboee0681f32012-03-06 12:14:42 +0100502 if (mean[0] > 999999.9) {
503 min[0] /= 1000.0;
504 max[0] /= 1000.0;
505 mean[0] /= 1000.0;
506 dev[0] /= 1000.0;
Jens Axboeca850992012-03-05 20:04:43 +0100507 bw_str = "MB";
508 }
509
Jens Axboe0b761302012-03-05 20:44:11 +0100510 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);
Jens Axboeca850992012-03-05 20:04:43 +0100513
Jens Axboe0b761302012-03-05 20:44:11 +0100514 box = gtk_hbox_new(FALSE, 3);
515 gtk_container_add(GTK_CONTAINER(frame), box);
Jens Axboeca850992012-03-05 20:04:43 +0100516
Jens Axboe0b761302012-03-05 20:44:11 +0100517 label = new_info_label_in_frame(box, "Minimum");
Jens Axboee0681f32012-03-06 12:14:42 +0100518 label_set_int_value(label, min[0]);
Jens Axboe0b761302012-03-05 20:44:11 +0100519 label = new_info_label_in_frame(box, "Maximum");
Jens Axboee0681f32012-03-06 12:14:42 +0100520 label_set_int_value(label, max[0]);
Jens Axboe0b761302012-03-05 20:44:11 +0100521 label = new_info_label_in_frame(box, "Percentage of jobs");
Jens Axboeca850992012-03-05 20:04:43 +0100522 sprintf(tmp, "%3.2f%%", p_of_agg);
523 gtk_label_set_text(GTK_LABEL(label), tmp);
Jens Axboe0b761302012-03-05 20:44:11 +0100524 label = new_info_label_in_frame(box, "Average");
Jens Axboee0681f32012-03-06 12:14:42 +0100525 sprintf(tmp, "%5.02f", mean[0]);
Jens Axboeca850992012-03-05 20:04:43 +0100526 gtk_label_set_text(GTK_LABEL(label), tmp);
Jens Axboe0b761302012-03-05 20:44:11 +0100527 label = new_info_label_in_frame(box, "Standard deviation");
Jens Axboee0681f32012-03-06 12:14:42 +0100528 sprintf(tmp, "%5.02f", dev[0]);
Jens Axboeca850992012-03-05 20:04:43 +0100529 gtk_label_set_text(GTK_LABEL(label), tmp);
530 }
531
Jens Axboee0681f32012-03-06 12:14:42 +0100532 if (calc_lat(&ts->slat_stat[ddir], &min[0], &max[0], &mean[0], &dev[0]))
Jens Axboe2b089892012-03-06 08:09:17 +0100533 flags |= GFIO_SLAT;
Jens Axboee0681f32012-03-06 12:14:42 +0100534 if (calc_lat(&ts->clat_stat[ddir], &min[1], &max[1], &mean[1], &dev[1]))
Jens Axboe2b089892012-03-06 08:09:17 +0100535 flags |= GFIO_CLAT;
Jens Axboee0681f32012-03-06 12:14:42 +0100536 if (calc_lat(&ts->lat_stat[ddir], &min[2], &max[2], &mean[2], &dev[2]))
Jens Axboe2b089892012-03-06 08:09:17 +0100537 flags |= GFIO_LAT;
538
539 if (flags) {
540 frame = gtk_frame_new("Latency");
541 gtk_box_pack_start(GTK_BOX(main_vbox), frame, FALSE, FALSE, 5);
542
543 vbox = gtk_vbox_new(FALSE, 3);
544 gtk_container_add(GTK_CONTAINER(frame), vbox);
545
546 if (flags & GFIO_SLAT)
Jens Axboee0681f32012-03-06 12:14:42 +0100547 gfio_show_lat(vbox, "Submission latency", min[0], max[0], mean[0], dev[0]);
Jens Axboe2b089892012-03-06 08:09:17 +0100548 if (flags & GFIO_CLAT)
Jens Axboee0681f32012-03-06 12:14:42 +0100549 gfio_show_lat(vbox, "Completion latency", min[1], max[1], mean[1], dev[1]);
Jens Axboe2b089892012-03-06 08:09:17 +0100550 if (flags & GFIO_LAT)
Jens Axboee0681f32012-03-06 12:14:42 +0100551 gfio_show_lat(vbox, "Total latency", min[2], max[2], mean[2], dev[2]);
Jens Axboe2b089892012-03-06 08:09:17 +0100552 }
553
554 if (ts->clat_percentiles)
555 gfio_show_clat_percentiles(main_vbox, ts, ddir);
556
557
Jens Axboe3650a3c2012-03-05 14:09:03 +0100558 free(io_p);
559 free(bw_p);
560 free(iops_p);
561}
562
Jens Axboee5bd1342012-03-05 21:38:12 +0100563static GtkWidget *gfio_output_lat_buckets(double *lat, unsigned int num,
564 const char **labels)
565{
566 GtkWidget *tree_view;
567 GtkTreeSelection *selection;
568 GtkListStore *model;
569 GtkTreeIter iter;
570 GType *types;
571 int i, skipped;
572
573 /*
574 * Check if all are empty, in which case don't bother
575 */
576 for (i = 0, skipped = 0; i < num; i++)
577 if (lat[i] <= 0.0)
578 skipped++;
579
580 if (skipped == num)
581 return NULL;
582
583 types = malloc(num * sizeof(GType));
584
585 for (i = 0; i < num; i++)
586 types[i] = G_TYPE_STRING;
587
588 model = gtk_list_store_newv(num, types);
589 free(types);
590 types = NULL;
591
592 tree_view = gtk_tree_view_new_with_model(GTK_TREE_MODEL(model));
593 gtk_widget_set_can_focus(tree_view, FALSE);
594
Jens Axboe661f7412012-03-06 13:55:45 +0100595 g_object_set(G_OBJECT(tree_view), "headers-visible", TRUE,
596 "enable-grid-lines", GTK_TREE_VIEW_GRID_LINES_BOTH, NULL);
597
Jens Axboee5bd1342012-03-05 21:38:12 +0100598 selection = gtk_tree_view_get_selection(GTK_TREE_VIEW(tree_view));
599 gtk_tree_selection_set_mode(GTK_TREE_SELECTION(selection), GTK_SELECTION_BROWSE);
600
601 for (i = 0; i < num; i++)
602 tree_view_column(tree_view, i, labels[i], ALIGN_RIGHT | UNSORTABLE);
603
604 gtk_list_store_append(model, &iter);
605
606 for (i = 0; i < num; i++) {
607 char fbuf[32];
608
609 if (lat[i] <= 0.0)
610 sprintf(fbuf, "0.00");
611 else
612 sprintf(fbuf, "%3.2f%%", lat[i]);
613
614 gtk_list_store_set(model, &iter, i, fbuf, -1);
615 }
616
617 return tree_view;
618}
619
620static void gfio_show_latency_buckets(GtkWidget *vbox, struct thread_stat *ts)
621{
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",
629 ">= 2000", };
630
631 stat_calc_lat_u(ts, io_u_lat_u);
632 stat_calc_lat_m(ts, io_u_lat_m);
633
634 tree_view = gfio_output_lat_buckets(io_u_lat_u, FIO_IO_U_LAT_U_NR, uranges);
635 if (tree_view) {
636 frame = gtk_frame_new("Latency buckets (usec)");
637 gtk_box_pack_start(GTK_BOX(vbox), frame, FALSE, FALSE, 5);
638
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);
642 }
643
644 tree_view = gfio_output_lat_buckets(io_u_lat_m, FIO_IO_U_LAT_M_NR, mranges);
645 if (tree_view) {
646 frame = gtk_frame_new("Latency buckets (msec)");
647 gtk_box_pack_start(GTK_BOX(vbox), frame, FALSE, FALSE, 5);
648
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);
652 }
653}
654
Jens Axboe2e331012012-03-05 22:07:54 +0100655static void gfio_show_cpu_usage(GtkWidget *vbox, struct thread_stat *ts)
656{
657 GtkWidget *box, *frame, *entry;
658 double usr_cpu, sys_cpu;
659 unsigned long runtime;
660 char tmp[32];
661
662 runtime = ts->total_run_time;
663 if (runtime) {
664 double runt = (double) runtime;
665
666 usr_cpu = (double) ts->usr_time * 100 / runt;
667 sys_cpu = (double) ts->sys_time * 100 / runt;
668 } else {
669 usr_cpu = 0;
670 sys_cpu = 0;
671 }
672
673 frame = gtk_frame_new("OS resources");
674 gtk_box_pack_start(GTK_BOX(vbox), frame, FALSE, FALSE, 5);
675
676 box = gtk_hbox_new(FALSE, 3);
677 gtk_container_add(GTK_CONTAINER(frame), box);
678
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);
691}
Jens Axboe19998db2012-03-06 09:17:59 +0100692static void gfio_add_sc_depths_tree(GtkListStore *model,
693 struct thread_stat *ts, unsigned int len,
694 int submit)
695{
696 double io_u_dist[FIO_IO_U_MAP_NR];
697 GtkTreeIter iter;
698 /* Bits 0, and 3-8 */
699 const int add_mask = 0x1f9;
700 int i, j;
701
702 if (submit)
703 stat_calc_dist(ts->io_u_submit, ts->total_submit, io_u_dist);
704 else
705 stat_calc_dist(ts->io_u_complete, ts->total_complete, io_u_dist);
706
707 gtk_list_store_append(model, &iter);
708
709 gtk_list_store_set(model, &iter, 0, submit ? "Submit" : "Complete", -1);
710
711 for (i = 1, j = 0; i < len; i++) {
712 char fbuf[32];
713
714 if (!(add_mask & (1UL << (i - 1))))
715 sprintf(fbuf, "0.0%%");
716 else {
717 sprintf(fbuf, "%3.1f%%", io_u_dist[j]);
718 j++;
719 }
720
721 gtk_list_store_set(model, &iter, i, fbuf, -1);
722 }
723
724}
725
726static void gfio_add_total_depths_tree(GtkListStore *model,
727 struct thread_stat *ts, unsigned int len)
728{
729 double io_u_dist[FIO_IO_U_MAP_NR];
730 GtkTreeIter iter;
731 /* Bits 1-6, and 8 */
732 const int add_mask = 0x17e;
733 int i, j;
734
735 stat_calc_dist(ts->io_u_map, ts_total_io_u(ts), io_u_dist);
736
737 gtk_list_store_append(model, &iter);
738
739 gtk_list_store_set(model, &iter, 0, "Total", -1);
740
741 for (i = 1, j = 0; i < len; i++) {
742 char fbuf[32];
743
744 if (!(add_mask & (1UL << (i - 1))))
745 sprintf(fbuf, "0.0%%");
746 else {
747 sprintf(fbuf, "%3.1f%%", io_u_dist[j]);
748 j++;
749 }
750
751 gtk_list_store_set(model, &iter, i, fbuf, -1);
752 }
753
754}
Jens Axboe2e331012012-03-05 22:07:54 +0100755
756static void gfio_show_io_depths(GtkWidget *vbox, struct thread_stat *ts)
757{
Jens Axboe2e331012012-03-05 22:07:54 +0100758 GtkWidget *frame, *box, *tree_view;
759 GtkTreeSelection *selection;
760 GtkListStore *model;
Jens Axboe2e331012012-03-05 22:07:54 +0100761 GType types[FIO_IO_U_MAP_NR + 1];
762 int i;
Jens Axboe19998db2012-03-06 09:17:59 +0100763#define NR_LABELS 10
764 const char *labels[NR_LABELS] = { "Depth", "0", "1", "2", "4", "8", "16", "32", "64", ">= 64" };
Jens Axboe2e331012012-03-05 22:07:54 +0100765
766 frame = gtk_frame_new("IO depths");
767 gtk_box_pack_start(GTK_BOX(vbox), frame, FALSE, FALSE, 5);
768
769 box = gtk_hbox_new(FALSE, 3);
770 gtk_container_add(GTK_CONTAINER(frame), box);
771
Jens Axboe19998db2012-03-06 09:17:59 +0100772 for (i = 0; i < NR_LABELS; i++)
Jens Axboe2e331012012-03-05 22:07:54 +0100773 types[i] = G_TYPE_STRING;
774
Jens Axboe19998db2012-03-06 09:17:59 +0100775 model = gtk_list_store_newv(NR_LABELS, types);
Jens Axboe2e331012012-03-05 22:07:54 +0100776
777 tree_view = gtk_tree_view_new_with_model(GTK_TREE_MODEL(model));
778 gtk_widget_set_can_focus(tree_view, FALSE);
779
Jens Axboe661f7412012-03-06 13:55:45 +0100780 g_object_set(G_OBJECT(tree_view), "headers-visible", TRUE,
781 "enable-grid-lines", GTK_TREE_VIEW_GRID_LINES_BOTH, NULL);
782
Jens Axboe2e331012012-03-05 22:07:54 +0100783 selection = gtk_tree_view_get_selection(GTK_TREE_VIEW(tree_view));
784 gtk_tree_selection_set_mode(GTK_TREE_SELECTION(selection), GTK_SELECTION_BROWSE);
785
Jens Axboe19998db2012-03-06 09:17:59 +0100786 for (i = 0; i < NR_LABELS; i++)
Jens Axboe2e331012012-03-05 22:07:54 +0100787 tree_view_column(tree_view, i, labels[i], ALIGN_RIGHT | UNSORTABLE);
788
Jens Axboe19998db2012-03-06 09:17:59 +0100789 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);
Jens Axboe2e331012012-03-05 22:07:54 +0100792
793 gtk_box_pack_start(GTK_BOX(box), tree_view, TRUE, FALSE, 3);
794}
795
Jens Axboef9d40b42012-03-06 09:52:49 +0100796static gboolean results_window_delete(GtkWidget *w, gpointer data)
797{
798 struct gui *ui = (struct gui *) data;
799
800 gtk_widget_destroy(w);
801 ui->results_window = NULL;
802 ui->results_notebook = NULL;
803 return TRUE;
804}
805
806static GtkWidget *get_results_window(struct gui *ui)
807{
808 GtkWidget *win, *notebook;
809
810 if (ui->results_window)
811 return ui->results_notebook;
812
813 win = gtk_window_new(GTK_WINDOW_TOPLEVEL);
814 gtk_window_set_title(GTK_WINDOW(win), "Results");
815 g_signal_connect(win, "delete-event", G_CALLBACK(results_window_delete), ui);
816 g_signal_connect(win, "destroy", G_CALLBACK(results_window_delete), ui);
817
818 notebook = gtk_notebook_new();
819 gtk_container_add(GTK_CONTAINER(win), notebook);
820
821 ui->results_window = win;
822 ui->results_notebook = notebook;
823 return ui->results_notebook;
824}
825
Jens Axboe3650a3c2012-03-05 14:09:03 +0100826static void gfio_display_ts(struct fio_client *client, struct thread_stat *ts,
827 struct group_run_stats *rs)
828{
Jens Axboef9d40b42012-03-06 09:52:49 +0100829 GtkWidget *res_win, *box, *vbox, *entry;
Jens Axboee0681f32012-03-06 12:14:42 +0100830 struct gfio_client *gc = client->client_data;
Jens Axboe3650a3c2012-03-05 14:09:03 +0100831
832 gdk_threads_enter();
833
Jens Axboee0681f32012-03-06 12:14:42 +0100834 res_win = get_results_window(gc->ui);
Jens Axboe3650a3c2012-03-05 14:09:03 +0100835
836 vbox = gtk_vbox_new(FALSE, 3);
Jens Axboe3650a3c2012-03-05 14:09:03 +0100837
838 box = gtk_hbox_new(TRUE, 3);
839 gtk_box_pack_start(GTK_BOX(vbox), box, FALSE, FALSE, 5);
840
Jens Axboef9d40b42012-03-06 09:52:49 +0100841 gtk_notebook_append_page(GTK_NOTEBOOK(res_win), vbox, gtk_label_new(ts->name));
842
Jens Axboee0681f32012-03-06 12:14:42 +0100843 gc->results_widget = vbox;
844
Jens Axboe3650a3c2012-03-05 14:09:03 +0100845 entry = new_info_entry_in_frame(box, "Name");
846 gtk_entry_set_text(GTK_ENTRY(entry), ts->name);
847 if (strlen(ts->description)) {
848 entry = new_info_entry_in_frame(box, "Description");
849 gtk_entry_set_text(GTK_ENTRY(entry), ts->description);
850 }
851 entry = new_info_entry_in_frame(box, "Group ID");
852 entry_set_int_value(entry, ts->groupid);
853 entry = new_info_entry_in_frame(box, "Jobs");
854 entry_set_int_value(entry, ts->members);
855 entry = new_info_entry_in_frame(box, "Error");
856 entry_set_int_value(entry, ts->error);
857 entry = new_info_entry_in_frame(box, "PID");
858 entry_set_int_value(entry, ts->pid);
859
860 if (ts->io_bytes[DDIR_READ])
861 gfio_show_ddir_status(vbox, rs, ts, DDIR_READ);
862 if (ts->io_bytes[DDIR_WRITE])
863 gfio_show_ddir_status(vbox, rs, ts, DDIR_WRITE);
864
Jens Axboee5bd1342012-03-05 21:38:12 +0100865 gfio_show_latency_buckets(vbox, ts);
Jens Axboe2e331012012-03-05 22:07:54 +0100866 gfio_show_cpu_usage(vbox, ts);
867 gfio_show_io_depths(vbox, ts);
Jens Axboee5bd1342012-03-05 21:38:12 +0100868
Jens Axboee0681f32012-03-06 12:14:42 +0100869 gtk_widget_show_all(gc->ui->results_window);
Jens Axboe3650a3c2012-03-05 14:09:03 +0100870 gdk_threads_leave();
871}
872
Jens Axboe084d1c62012-03-03 20:28:07 +0100873static void gfio_text_op(struct fio_client *client, struct fio_net_cmd *cmd)
Stephen M. Camerona1820202012-02-24 08:17:31 +0100874{
Jens Axboe9b260bd2012-03-06 11:02:52 +0100875 struct cmd_text_pdu *p = (struct cmd_text_pdu *) cmd->payload;
Jens Axboee0681f32012-03-06 12:14:42 +0100876 struct gfio_client *gc = client->client_data;
Jens Axboe9b260bd2012-03-06 11:02:52 +0100877 GtkTreeIter iter;
878 struct tm *tm;
879 time_t sec;
880 char tmp[64], timebuf[80];
Stephen M. Cameron736f2df2012-02-24 08:17:32 +0100881
Jens Axboe9b260bd2012-03-06 11:02:52 +0100882 sec = p->log_sec;
883 tm = localtime(&sec);
884 strftime(tmp, sizeof(tmp), "%Y-%m-%d %H:%M:%S", tm);
885 sprintf(timebuf, "%s.%03ld", tmp, p->log_usec / 1000);
886
Stephen M. Cameron736f2df2012-02-24 08:17:32 +0100887 gdk_threads_enter();
Jens Axboe9b260bd2012-03-06 11:02:52 +0100888
Jens Axboee0681f32012-03-06 12:14:42 +0100889 gtk_list_store_append(gc->ui->log_model, &iter);
890 gtk_list_store_set(gc->ui->log_model, &iter, 0, timebuf, -1);
891 gtk_list_store_set(gc->ui->log_model, &iter, 1, client->hostname, -1);
892 gtk_list_store_set(gc->ui->log_model, &iter, 2, p->level, -1);
893 gtk_list_store_set(gc->ui->log_model, &iter, 3, p->buf, -1);
Jens Axboe9b260bd2012-03-06 11:02:52 +0100894
Stephen M. Cameron736f2df2012-02-24 08:17:32 +0100895 gdk_threads_leave();
Stephen M. Camerona1820202012-02-24 08:17:31 +0100896}
897
898static void gfio_disk_util_op(struct fio_client *client, struct fio_net_cmd *cmd)
899{
Jens Axboee0681f32012-03-06 12:14:42 +0100900 struct cmd_du_pdu *p = (struct cmd_du_pdu *) cmd->payload;
901 struct gfio_client *gc = client->client_data;
902 GtkWidget *box, *frame, *entry, *vbox;
Jens Axboe604cfe32012-03-07 19:51:36 +0100903 double util;
904 char tmp[16];
Jens Axboee0681f32012-03-06 12:14:42 +0100905
Jens Axboe0050e5f2012-03-06 09:23:27 +0100906 gdk_threads_enter();
Jens Axboee0681f32012-03-06 12:14:42 +0100907
Jens Axboe45dcb2e2012-03-07 16:16:50 +0100908 if (!gc->results_widget)
Jens Axboee0681f32012-03-06 12:14:42 +0100909 goto out;
Jens Axboee0681f32012-03-06 12:14:42 +0100910
911 if (!gc->disk_util_frame) {
912 gc->disk_util_frame = gtk_frame_new("Disk utilization");
913 gtk_box_pack_start(GTK_BOX(gc->results_widget), gc->disk_util_frame, FALSE, FALSE, 5);
914 }
915
916 vbox = gtk_vbox_new(FALSE, 3);
917 gtk_container_add(GTK_CONTAINER(gc->disk_util_frame), vbox);
918
919 frame = gtk_frame_new((char *) p->dus.name);
920 gtk_box_pack_start(GTK_BOX(vbox), frame, FALSE, FALSE, 2);
921
922 box = gtk_vbox_new(FALSE, 3);
923 gtk_container_add(GTK_CONTAINER(frame), box);
924
925 frame = gtk_frame_new("Read");
926 gtk_box_pack_start(GTK_BOX(box), frame, FALSE, FALSE, 2);
927 vbox = gtk_hbox_new(TRUE, 3);
928 gtk_container_add(GTK_CONTAINER(frame), vbox);
929 entry = new_info_entry_in_frame(vbox, "IOs");
930 entry_set_int_value(entry, p->dus.ios[0]);
931 entry = new_info_entry_in_frame(vbox, "Merges");
932 entry_set_int_value(entry, p->dus.merges[0]);
933 entry = new_info_entry_in_frame(vbox, "Sectors");
934 entry_set_int_value(entry, p->dus.sectors[0]);
935 entry = new_info_entry_in_frame(vbox, "Ticks");
936 entry_set_int_value(entry, p->dus.ticks[0]);
937
938 frame = gtk_frame_new("Write");
939 gtk_box_pack_start(GTK_BOX(box), frame, FALSE, FALSE, 2);
940 vbox = gtk_hbox_new(TRUE, 3);
941 gtk_container_add(GTK_CONTAINER(frame), vbox);
942 entry = new_info_entry_in_frame(vbox, "IOs");
943 entry_set_int_value(entry, p->dus.ios[1]);
944 entry = new_info_entry_in_frame(vbox, "Merges");
945 entry_set_int_value(entry, p->dus.merges[1]);
946 entry = new_info_entry_in_frame(vbox, "Sectors");
947 entry_set_int_value(entry, p->dus.sectors[1]);
948 entry = new_info_entry_in_frame(vbox, "Ticks");
949 entry_set_int_value(entry, p->dus.ticks[1]);
950
951 frame = gtk_frame_new("Shared");
952 gtk_box_pack_start(GTK_BOX(box), frame, FALSE, FALSE, 2);
953 vbox = gtk_hbox_new(TRUE, 3);
954 gtk_container_add(GTK_CONTAINER(frame), vbox);
955 entry = new_info_entry_in_frame(vbox, "IO ticks");
956 entry_set_int_value(entry, p->dus.io_ticks);
957 entry = new_info_entry_in_frame(vbox, "Time in queue");
958 entry_set_int_value(entry, p->dus.time_in_queue);
959
Jens Axboe604cfe32012-03-07 19:51:36 +0100960 util = 0.0;
961 if (p->dus.msec)
962 util = (double) 100 * p->dus.io_ticks / (double) p->dus.msec;
963 if (util > 100.0)
964 util = 100.0;
965
966 sprintf(tmp, "%3.2f%%", util);
967 entry = new_info_entry_in_frame(vbox, "Disk utilization");
968 gtk_entry_set_text(GTK_ENTRY(entry), tmp);
969
Jens Axboee0681f32012-03-06 12:14:42 +0100970 gtk_widget_show_all(gc->results_widget);
971out:
Jens Axboe0050e5f2012-03-06 09:23:27 +0100972 gdk_threads_leave();
Stephen M. Camerona1820202012-02-24 08:17:31 +0100973}
974
Jens Axboe3650a3c2012-03-05 14:09:03 +0100975extern int sum_stat_clients;
976extern struct thread_stat client_ts;
977extern struct group_run_stats client_gs;
978
979static int sum_stat_nr;
980
Jens Axboe89e5fad2012-03-05 09:21:12 +0100981static void gfio_thread_status_op(struct fio_client *client,
982 struct fio_net_cmd *cmd)
Stephen M. Camerona1820202012-02-24 08:17:31 +0100983{
Jens Axboe3650a3c2012-03-05 14:09:03 +0100984 struct cmd_ts_pdu *p = (struct cmd_ts_pdu *) cmd->payload;
985
986 gfio_display_ts(client, &p->ts, &p->rs);
987
988 if (sum_stat_clients == 1)
989 return;
990
991 sum_thread_stats(&client_ts, &p->ts, sum_stat_nr);
992 sum_group_stats(&client_gs, &p->rs);
993
994 client_ts.members++;
995 client_ts.groupid = p->ts.groupid;
996
997 if (++sum_stat_nr == sum_stat_clients) {
998 strcpy(client_ts.name, "All clients");
999 gfio_display_ts(client, &client_ts, &client_gs);
1000 }
Stephen M. Camerona1820202012-02-24 08:17:31 +01001001}
1002
Jens Axboe89e5fad2012-03-05 09:21:12 +01001003static void gfio_group_stats_op(struct fio_client *client,
1004 struct fio_net_cmd *cmd)
Stephen M. Camerona1820202012-02-24 08:17:31 +01001005{
Jens Axboe0050e5f2012-03-06 09:23:27 +01001006 gdk_threads_enter();
Stephen M. Camerona1820202012-02-24 08:17:31 +01001007 printf("gfio_group_stats_op called\n");
Jens Axboe89e5fad2012-03-05 09:21:12 +01001008 fio_client_ops.group_stats(client, cmd);
Jens Axboe0050e5f2012-03-06 09:23:27 +01001009 gdk_threads_leave();
Stephen M. Camerona1820202012-02-24 08:17:31 +01001010}
1011
Stephen M. Cameron3ea48b82012-03-07 19:40:58 +01001012static gint on_config_drawing_area(GtkWidget *w, GdkEventConfigure *event)
1013{
1014 ui.drawing_area_xdim = w->allocation.width;
1015 ui.drawing_area_ydim = w->allocation.height;
1016 return TRUE;
1017}
1018
Jens Axboe2fd3bb02012-03-07 08:07:39 +01001019static int on_expose_drawing_area(GtkWidget *w, GdkEvent *event, gpointer p)
1020{
1021 struct gui *ui = (struct gui *) p;
1022 cairo_t *cr;
1023
Stephen M. Cameron3ea48b82012-03-07 19:40:58 +01001024 graph_set_size(ui->iops_graph, ui->drawing_area_xdim / 2.0,
1025 ui->drawing_area_ydim);
1026 graph_set_size(ui->bandwidth_graph, ui->drawing_area_xdim / 2.0,
1027 ui->drawing_area_ydim);
Jens Axboe2fd3bb02012-03-07 08:07:39 +01001028 cr = gdk_cairo_create(w->window);
1029
1030 cairo_set_source_rgb(cr, 0, 0, 0);
1031
1032 cairo_save(cr);
1033 cairo_translate(cr, 0, 0);
1034 line_graph_draw(ui->bandwidth_graph, cr);
1035 cairo_stroke(cr);
1036 cairo_restore(cr);
1037
1038 cairo_save(cr);
Stephen M. Cameron3ea48b82012-03-07 19:40:58 +01001039 cairo_translate(cr, ui->drawing_area_xdim / 2.0, 0);
Jens Axboe2fd3bb02012-03-07 08:07:39 +01001040 line_graph_draw(ui->iops_graph, cr);
1041 cairo_stroke(cr);
1042 cairo_restore(cr);
1043 cairo_destroy(cr);
1044
1045 return FALSE;
1046}
1047
Jens Axboe3e47bd22012-02-29 13:45:02 +01001048static void gfio_update_eta(struct jobs_eta *je)
1049{
1050 static int eta_good;
1051 char eta_str[128];
1052 char output[256];
1053 char tmp[32];
1054 double perc = 0.0;
1055 int i2p = 0;
1056
Jens Axboe0050e5f2012-03-06 09:23:27 +01001057 gdk_threads_enter();
1058
Jens Axboe3e47bd22012-02-29 13:45:02 +01001059 eta_str[0] = '\0';
1060 output[0] = '\0';
1061
1062 if (je->eta_sec != INT_MAX && je->elapsed_sec) {
1063 perc = (double) je->elapsed_sec / (double) (je->elapsed_sec + je->eta_sec);
1064 eta_to_str(eta_str, je->eta_sec);
1065 }
1066
1067 sprintf(tmp, "%u", je->nr_running);
Jens Axboeca850992012-03-05 20:04:43 +01001068 gtk_entry_set_text(GTK_ENTRY(ui.eta.jobs), tmp);
Jens Axboe3e47bd22012-02-29 13:45:02 +01001069 sprintf(tmp, "%u", je->files_open);
Jens Axboeca850992012-03-05 20:04:43 +01001070 gtk_entry_set_text(GTK_ENTRY(ui.eta.files), tmp);
Jens Axboe3e47bd22012-02-29 13:45:02 +01001071
1072#if 0
1073 if (je->m_rate[0] || je->m_rate[1] || je->t_rate[0] || je->t_rate[1]) {
1074 if (je->m_rate || je->t_rate) {
1075 char *tr, *mr;
1076
1077 mr = num2str(je->m_rate, 4, 0, i2p);
1078 tr = num2str(je->t_rate, 4, 0, i2p);
Jens Axboeca850992012-03-05 20:04:43 +01001079 gtk_entry_set_text(GTK_ENTRY(ui.eta);
Jens Axboe3e47bd22012-02-29 13:45:02 +01001080 p += sprintf(p, ", CR=%s/%s KB/s", tr, mr);
1081 free(tr);
1082 free(mr);
1083 } else if (je->m_iops || je->t_iops)
1084 p += sprintf(p, ", CR=%d/%d IOPS", je->t_iops, je->m_iops);
Jens Axboeebbd89c2012-03-02 11:21:13 +01001085
Jens Axboeca850992012-03-05 20:04:43 +01001086 gtk_entry_set_text(GTK_ENTRY(ui.eta.cr_bw), "---");
1087 gtk_entry_set_text(GTK_ENTRY(ui.eta.cr_iops), "---");
1088 gtk_entry_set_text(GTK_ENTRY(ui.eta.cw_bw), "---");
1089 gtk_entry_set_text(GTK_ENTRY(ui.eta.cw_iops), "---");
Jens Axboe3e47bd22012-02-29 13:45:02 +01001090#endif
1091
1092 if (je->eta_sec != INT_MAX && je->nr_running) {
1093 char *iops_str[2];
1094 char *rate_str[2];
1095
1096 if ((!je->eta_sec && !eta_good) || je->nr_ramp == je->nr_running)
1097 strcpy(output, "-.-% done");
1098 else {
1099 eta_good = 1;
1100 perc *= 100.0;
1101 sprintf(output, "%3.1f%% done", perc);
1102 }
1103
1104 rate_str[0] = num2str(je->rate[0], 5, 10, i2p);
1105 rate_str[1] = num2str(je->rate[1], 5, 10, i2p);
1106
1107 iops_str[0] = num2str(je->iops[0], 4, 1, 0);
1108 iops_str[1] = num2str(je->iops[1], 4, 1, 0);
1109
Jens Axboeca850992012-03-05 20:04:43 +01001110 gtk_entry_set_text(GTK_ENTRY(ui.eta.read_bw), rate_str[0]);
1111 gtk_entry_set_text(GTK_ENTRY(ui.eta.read_iops), iops_str[0]);
1112 gtk_entry_set_text(GTK_ENTRY(ui.eta.write_bw), rate_str[1]);
1113 gtk_entry_set_text(GTK_ENTRY(ui.eta.write_iops), iops_str[1]);
Jens Axboe3e47bd22012-02-29 13:45:02 +01001114
Jens Axboe2fd3bb02012-03-07 08:07:39 +01001115 graph_add_xy_data(ui.iops_graph, "Read IOPS", je->elapsed_sec, je->iops[0]);
1116 graph_add_xy_data(ui.iops_graph, "Write IOPS", je->elapsed_sec, je->iops[1]);
1117 graph_add_xy_data(ui.bandwidth_graph, "Read Bandwidth", je->elapsed_sec, je->rate[0]);
1118 graph_add_xy_data(ui.bandwidth_graph, "Write Bandwidth", je->elapsed_sec, je->rate[1]);
1119
Jens Axboe3e47bd22012-02-29 13:45:02 +01001120 free(rate_str[0]);
1121 free(rate_str[1]);
1122 free(iops_str[0]);
1123 free(iops_str[1]);
1124 }
1125
1126 if (eta_str[0]) {
1127 char *dst = output + strlen(output);
1128
1129 sprintf(dst, " - %s", eta_str);
1130 }
1131
1132 gfio_update_thread_status(output, perc);
Jens Axboe0050e5f2012-03-06 09:23:27 +01001133 gdk_threads_leave();
Jens Axboe3e47bd22012-02-29 13:45:02 +01001134}
1135
Stephen M. Camerona1820202012-02-24 08:17:31 +01001136static void gfio_probe_op(struct fio_client *client, struct fio_net_cmd *cmd)
1137{
Jens Axboe843ad232012-02-29 11:44:53 +01001138 struct cmd_probe_pdu *probe = (struct cmd_probe_pdu *) cmd->payload;
Jens Axboe88f6e7a2012-03-06 12:55:29 +01001139 struct gfio_client *gc = client->client_data;
1140 struct gui *ui = gc->ui;
Jens Axboe843ad232012-02-29 11:44:53 +01001141 const char *os, *arch;
1142 char buf[64];
1143
1144 os = fio_get_os_string(probe->os);
1145 if (!os)
1146 os = "unknown";
1147
1148 arch = fio_get_arch_string(probe->arch);
1149 if (!arch)
1150 os = "unknown";
1151
1152 if (!client->name)
1153 client->name = strdup((char *) probe->hostname);
1154
Jens Axboe0050e5f2012-03-06 09:23:27 +01001155 gdk_threads_enter();
1156
Jens Axboe88f6e7a2012-03-06 12:55:29 +01001157 gtk_label_set_text(GTK_LABEL(ui->probe.hostname), (char *) probe->hostname);
1158 gtk_label_set_text(GTK_LABEL(ui->probe.os), os);
1159 gtk_label_set_text(GTK_LABEL(ui->probe.arch), arch);
Jens Axboe843ad232012-02-29 11:44:53 +01001160 sprintf(buf, "%u.%u.%u", probe->fio_major, probe->fio_minor, probe->fio_patch);
Jens Axboe88f6e7a2012-03-06 12:55:29 +01001161 gtk_label_set_text(GTK_LABEL(ui->probe.fio_ver), buf);
1162
1163 gfio_set_connected(ui, 1);
Jens Axboe0050e5f2012-03-06 09:23:27 +01001164
1165 gdk_threads_leave();
Stephen M. Camerona1820202012-02-24 08:17:31 +01001166}
1167
Stephen M. Cameron04cc6b72012-02-24 08:17:31 +01001168static void gfio_update_thread_status(char *status_message, double perc)
Stephen M. Cameron5b7573a2012-02-24 08:17:31 +01001169{
1170 static char message[100];
1171 const char *m = message;
1172
1173 strncpy(message, status_message, sizeof(message) - 1);
Stephen M. Cameron04cc6b72012-02-24 08:17:31 +01001174 gtk_progress_bar_set_text(
1175 GTK_PROGRESS_BAR(ui.thread_status_pb), m);
1176 gtk_progress_bar_set_fraction(
1177 GTK_PROGRESS_BAR(ui.thread_status_pb), perc / 100.0);
Stephen M. Cameron5b7573a2012-02-24 08:17:31 +01001178 gtk_widget_queue_draw(ui.window);
Stephen M. Cameron5b7573a2012-02-24 08:17:31 +01001179}
1180
Jens Axboe3ec62ec2012-03-01 12:01:29 +01001181static void gfio_quit_op(struct fio_client *client)
1182{
Jens Axboee0681f32012-03-06 12:14:42 +01001183 struct gfio_client *gc = client->client_data;
Jens Axboe3ec62ec2012-03-01 12:01:29 +01001184
Jens Axboe0050e5f2012-03-06 09:23:27 +01001185 gdk_threads_enter();
Jens Axboee0681f32012-03-06 12:14:42 +01001186 gfio_set_connected(gc->ui, 0);
Jens Axboe0050e5f2012-03-06 09:23:27 +01001187 gdk_threads_leave();
Jens Axboe3ec62ec2012-03-01 12:01:29 +01001188}
1189
Jens Axboe807f9972012-03-02 10:25:24 +01001190static void gfio_add_job_op(struct fio_client *client, struct fio_net_cmd *cmd)
1191{
1192 struct cmd_add_job_pdu *p = (struct cmd_add_job_pdu *) cmd->payload;
Jens Axboee0681f32012-03-06 12:14:42 +01001193 struct gfio_client *gc = client->client_data;
1194 struct gui *ui = gc->ui;
Jens Axboe807f9972012-03-02 10:25:24 +01001195 char tmp[8];
1196 int i;
1197
1198 p->iodepth = le32_to_cpu(p->iodepth);
1199 p->rw = le32_to_cpu(p->rw);
1200
1201 for (i = 0; i < 2; i++) {
1202 p->min_bs[i] = le32_to_cpu(p->min_bs[i]);
1203 p->max_bs[i] = le32_to_cpu(p->max_bs[i]);
1204 }
1205
1206 p->numjobs = le32_to_cpu(p->numjobs);
1207 p->group_reporting = le32_to_cpu(p->group_reporting);
1208
Jens Axboe0050e5f2012-03-06 09:23:27 +01001209 gdk_threads_enter();
1210
Jens Axboeca850992012-03-05 20:04:43 +01001211 gtk_entry_set_text(GTK_ENTRY(ui->eta.name), (gchar *) p->jobname);
1212 gtk_entry_set_text(GTK_ENTRY(ui->eta.iotype), ddir_str(p->rw));
1213 gtk_entry_set_text(GTK_ENTRY(ui->eta.ioengine), (gchar *) p->ioengine);
Jens Axboe807f9972012-03-02 10:25:24 +01001214
1215 sprintf(tmp, "%u", p->iodepth);
Jens Axboeca850992012-03-05 20:04:43 +01001216 gtk_entry_set_text(GTK_ENTRY(ui->eta.iodepth), tmp);
Jens Axboe0050e5f2012-03-06 09:23:27 +01001217
1218 gdk_threads_leave();
Jens Axboe807f9972012-03-02 10:25:24 +01001219}
1220
Jens Axboeed727a42012-03-02 12:14:40 +01001221static void gfio_client_timed_out(struct fio_client *client)
1222{
Jens Axboee0681f32012-03-06 12:14:42 +01001223 struct gfio_client *gc = client->client_data;
Jens Axboeed727a42012-03-02 12:14:40 +01001224 GtkWidget *dialog, *label, *content;
1225 char buf[256];
1226
1227 gdk_threads_enter();
1228
Jens Axboee0681f32012-03-06 12:14:42 +01001229 gfio_set_connected(gc->ui, 0);
1230 clear_ui_info(gc->ui);
Jens Axboeed727a42012-03-02 12:14:40 +01001231
1232 sprintf(buf, "Client %s: timeout talking to server.\n", client->hostname);
1233
1234 dialog = gtk_dialog_new_with_buttons("Timed out!",
Jens Axboee0681f32012-03-06 12:14:42 +01001235 GTK_WINDOW(gc->ui->window),
Jens Axboeed727a42012-03-02 12:14:40 +01001236 GTK_DIALOG_MODAL | GTK_DIALOG_DESTROY_WITH_PARENT,
1237 GTK_STOCK_OK, GTK_RESPONSE_OK, NULL);
1238
Jens Axboef1299092012-03-07 20:00:02 +01001239 /* gtk_dialog_get_content_area() is 2.14 and newer */
1240 content = GTK_DIALOG(dialog)->vbox;
1241
Jens Axboeed727a42012-03-02 12:14:40 +01001242 label = gtk_label_new((const gchar *) buf);
1243 gtk_container_add(GTK_CONTAINER(content), label);
1244 gtk_widget_show_all(dialog);
1245 gtk_dialog_set_default_response(GTK_DIALOG(dialog), GTK_RESPONSE_ACCEPT);
1246
1247 gtk_dialog_run(GTK_DIALOG(dialog));
1248 gtk_widget_destroy(dialog);
1249
1250 gdk_threads_leave();
1251}
1252
Stephen M. Camerona1820202012-02-24 08:17:31 +01001253struct client_ops gfio_client_ops = {
Jens Axboe0420ba62012-02-29 11:16:52 +01001254 .text_op = gfio_text_op,
1255 .disk_util = gfio_disk_util_op,
1256 .thread_status = gfio_thread_status_op,
1257 .group_stats = gfio_group_stats_op,
Jens Axboea5276612012-03-04 15:15:08 +01001258 .eta = gfio_update_eta,
Jens Axboe0420ba62012-02-29 11:16:52 +01001259 .probe = gfio_probe_op,
Jens Axboe3ec62ec2012-03-01 12:01:29 +01001260 .quit = gfio_quit_op,
Jens Axboe807f9972012-03-02 10:25:24 +01001261 .add_job = gfio_add_job_op,
Jens Axboeed727a42012-03-02 12:14:40 +01001262 .timed_out = gfio_client_timed_out,
Jens Axboe3ec62ec2012-03-01 12:01:29 +01001263 .stay_connected = 1,
Stephen M. Camerona1820202012-02-24 08:17:31 +01001264};
1265
Stephen M. Cameronff1f3282012-02-24 08:17:30 +01001266static void quit_clicked(__attribute__((unused)) GtkWidget *widget,
1267 __attribute__((unused)) gpointer data)
1268{
1269 gtk_main_quit();
1270}
1271
Stephen M. Cameron25927252012-02-24 08:17:31 +01001272static void *job_thread(void *arg)
Stephen M. Cameronf3074002012-02-24 08:17:30 +01001273{
Stephen M. Cameron25927252012-02-24 08:17:31 +01001274 fio_handle_clients(&gfio_client_ops);
Stephen M. Cameron25927252012-02-24 08:17:31 +01001275 return NULL;
1276}
1277
Jens Axboe0420ba62012-02-29 11:16:52 +01001278static int send_job_files(struct gui *ui)
Stephen M. Cameron60f6b332012-02-24 08:17:32 +01001279{
Jens Axboe441013b2012-03-01 08:01:52 +01001280 int i, ret = 0;
Stephen M. Cameron60f6b332012-02-24 08:17:32 +01001281
Jens Axboe0420ba62012-02-29 11:16:52 +01001282 for (i = 0; i < ui->nr_job_files; i++) {
1283 ret = fio_clients_send_ini(ui->job_files[i]);
Jens Axboe441013b2012-03-01 08:01:52 +01001284 if (ret)
1285 break;
1286
Jens Axboe0420ba62012-02-29 11:16:52 +01001287 free(ui->job_files[i]);
1288 ui->job_files[i] = NULL;
Jens Axboe441013b2012-03-01 08:01:52 +01001289 }
1290 while (i < ui->nr_job_files) {
1291 free(ui->job_files[i]);
1292 ui->job_files[i] = NULL;
1293 i++;
Jens Axboe0420ba62012-02-29 11:16:52 +01001294 }
1295
Jens Axboe441013b2012-03-01 08:01:52 +01001296 return ret;
Stephen M. Cameron60f6b332012-02-24 08:17:32 +01001297}
1298
Jens Axboe3ec62ec2012-03-01 12:01:29 +01001299static void start_job_thread(struct gui *ui)
Stephen M. Cameron25927252012-02-24 08:17:31 +01001300{
Jens Axboe0420ba62012-02-29 11:16:52 +01001301 if (send_job_files(ui)) {
Stephen M. Cameron60f6b332012-02-24 08:17:32 +01001302 printf("Yeah, I didn't really like those options too much.\n");
Stephen M. Cameron60f6b332012-02-24 08:17:32 +01001303 gtk_widget_set_sensitive(ui->button[START_JOB_BUTTON], 1);
1304 return;
1305 }
Stephen M. Cameron25927252012-02-24 08:17:31 +01001306}
1307
Jens Axboe63a130b2012-03-06 20:08:59 +01001308static void *server_thread(void *arg)
1309{
1310 is_backend = 1;
1311 gfio_server_running = 1;
1312 fio_start_server(NULL);
1313 gfio_server_running = 0;
1314 return NULL;
1315}
1316
1317static void gfio_start_server(struct gui *ui)
1318{
1319 if (!gfio_server_running) {
1320 gfio_server_running = 1;
1321 pthread_create(&ui->server_t, NULL, server_thread, NULL);
Jens Axboee34f6ad2012-03-06 20:47:15 +01001322 pthread_detach(ui->server_t);
Jens Axboe63a130b2012-03-06 20:08:59 +01001323 }
1324}
1325
Stephen M. Cameron25927252012-02-24 08:17:31 +01001326static void start_job_clicked(__attribute__((unused)) GtkWidget *widget,
1327 gpointer data)
1328{
1329 struct gui *ui = data;
1330
Stephen M. Cameron25927252012-02-24 08:17:31 +01001331 gtk_widget_set_sensitive(ui->button[START_JOB_BUTTON], 0);
Jens Axboe3ec62ec2012-03-01 12:01:29 +01001332 start_job_thread(ui);
Stephen M. Cameronf3074002012-02-24 08:17:30 +01001333}
1334
Jens Axboedf06f222012-03-02 13:32:04 +01001335static void file_open(GtkWidget *w, gpointer data);
1336
1337static void connect_clicked(GtkWidget *widget, gpointer data)
Jens Axboe3e47bd22012-02-29 13:45:02 +01001338{
Jens Axboe3ec62ec2012-03-01 12:01:29 +01001339 struct gui *ui = data;
1340
1341 if (!ui->connected) {
Jens Axboedf06f222012-03-02 13:32:04 +01001342 if (!ui->nr_job_files)
1343 file_open(widget, data);
Jens Axboe8663ea62012-03-02 14:04:30 +01001344 gtk_progress_bar_set_text(GTK_PROGRESS_BAR(ui->thread_status_pb), "No jobs running");
Jens Axboee34f6ad2012-03-06 20:47:15 +01001345 gtk_progress_bar_set_fraction(GTK_PROGRESS_BAR(ui->thread_status_pb), 0.0);
Jens Axboe69406b92012-03-06 14:00:42 +01001346 if (!fio_clients_connect()) {
1347 pthread_create(&ui->t, NULL, job_thread, NULL);
1348 gtk_widget_set_sensitive(ui->button[CONNECT_BUTTON], 0);
1349 }
Jens Axboedf06f222012-03-02 13:32:04 +01001350 } else {
1351 fio_clients_terminate();
Jens Axboe3ec62ec2012-03-01 12:01:29 +01001352 gfio_set_connected(ui, 0);
Jens Axboe88432652012-03-02 19:09:31 +01001353 clear_ui_info(ui);
Jens Axboedf06f222012-03-02 13:32:04 +01001354 }
Jens Axboe3e47bd22012-02-29 13:45:02 +01001355}
1356
Stephen M. Cameronf3074002012-02-24 08:17:30 +01001357static void add_button(struct gui *ui, int i, GtkWidget *buttonbox,
1358 struct button_spec *buttonspec)
1359{
1360 ui->button[i] = gtk_button_new_with_label(buttonspec->buttontext);
1361 g_signal_connect(ui->button[i], "clicked", G_CALLBACK (buttonspec->f), ui);
Jens Axboe3ec62ec2012-03-01 12:01:29 +01001362 gtk_box_pack_start(GTK_BOX (ui->buttonbox), ui->button[i], FALSE, FALSE, 3);
Stephen M. Cameronf3074002012-02-24 08:17:30 +01001363 gtk_widget_set_tooltip_text(ui->button[i], buttonspeclist[i].tooltiptext);
Jens Axboe3e47bd22012-02-29 13:45:02 +01001364 gtk_widget_set_sensitive(ui->button[i], !buttonspec->start_insensitive);
Stephen M. Cameronf3074002012-02-24 08:17:30 +01001365}
1366
1367static void add_buttons(struct gui *ui,
1368 struct button_spec *buttonlist,
1369 int nbuttons)
1370{
1371 int i;
1372
Stephen M. Cameronf3074002012-02-24 08:17:30 +01001373 for (i = 0; i < nbuttons; i++)
1374 add_button(ui, i, ui->buttonbox, &buttonlist[i]);
1375}
1376
Jens Axboe0420ba62012-02-29 11:16:52 +01001377static void on_info_bar_response(GtkWidget *widget, gint response,
1378 gpointer data)
1379{
1380 if (response == GTK_RESPONSE_OK) {
1381 gtk_widget_destroy(widget);
1382 ui.error_info_bar = NULL;
1383 }
1384}
1385
Jens Axboedf06f222012-03-02 13:32:04 +01001386void report_error(GError *error)
Jens Axboe0420ba62012-02-29 11:16:52 +01001387{
1388 if (ui.error_info_bar == NULL) {
1389 ui.error_info_bar = gtk_info_bar_new_with_buttons(GTK_STOCK_OK,
1390 GTK_RESPONSE_OK,
1391 NULL);
1392 g_signal_connect(ui.error_info_bar, "response", G_CALLBACK(on_info_bar_response), NULL);
1393 gtk_info_bar_set_message_type(GTK_INFO_BAR(ui.error_info_bar),
1394 GTK_MESSAGE_ERROR);
1395
1396 ui.error_label = gtk_label_new(error->message);
1397 GtkWidget *container = gtk_info_bar_get_content_area(GTK_INFO_BAR(ui.error_info_bar));
1398 gtk_container_add(GTK_CONTAINER(container), ui.error_label);
1399
1400 gtk_box_pack_start(GTK_BOX(ui.vbox), ui.error_info_bar, FALSE, FALSE, 0);
1401 gtk_widget_show_all(ui.vbox);
1402 } else {
1403 char buffer[256];
1404 snprintf(buffer, sizeof(buffer), "Failed to open file.");
1405 gtk_label_set(GTK_LABEL(ui.error_label), buffer);
1406 }
1407}
1408
Jens Axboe62bc9372012-03-07 11:45:07 +01001409struct connection_widgets
1410{
1411 GtkWidget *hentry;
1412 GtkWidget *combo;
1413 GtkWidget *button;
1414};
1415
1416static void hostname_cb(GtkEntry *entry, gpointer data)
1417{
1418 struct connection_widgets *cw = data;
1419 int uses_net = 0, is_localhost = 0;
1420 const gchar *text;
1421 gchar *ctext;
1422
1423 /*
1424 * Check whether to display the 'auto start backend' box
1425 * or not. Show it if we are a localhost and using network,
1426 * or using a socket.
1427 */
1428 ctext = gtk_combo_box_get_active_text(GTK_COMBO_BOX(cw->combo));
1429 if (!ctext || !strncmp(ctext, "IPv4", 4) || !strncmp(ctext, "IPv6", 4))
1430 uses_net = 1;
1431 g_free(ctext);
1432
1433 if (uses_net) {
1434 text = gtk_entry_get_text(GTK_ENTRY(cw->hentry));
1435 if (!strcmp(text, "127.0.0.1") || !strcmp(text, "localhost") ||
1436 !strcmp(text, "::1") || !strcmp(text, "ip6-localhost") ||
1437 !strcmp(text, "ip6-loopback"))
1438 is_localhost = 1;
1439 }
1440
1441 if (!uses_net || is_localhost) {
1442 gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(cw->button), 1);
1443 gtk_widget_set_sensitive(cw->button, 1);
1444 } else {
1445 gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(cw->button), 0);
1446 gtk_widget_set_sensitive(cw->button, 0);
1447 }
1448}
1449
Jens Axboeb9f3c7e2012-03-02 14:27:17 +01001450static int get_connection_details(char **host, int *port, int *type,
1451 int *server_start)
Jens Axboea7a42ce2012-03-02 13:12:04 +01001452{
Jens Axboe62bc9372012-03-07 11:45:07 +01001453 GtkWidget *dialog, *box, *vbox, *hbox, *frame, *pentry;
1454 struct connection_widgets cw;
Jens Axboea7a42ce2012-03-02 13:12:04 +01001455 char *typeentry;
1456
1457 dialog = gtk_dialog_new_with_buttons("Connection details",
1458 GTK_WINDOW(ui.window),
1459 GTK_DIALOG_DESTROY_WITH_PARENT,
1460 GTK_STOCK_OK, GTK_RESPONSE_ACCEPT,
1461 GTK_STOCK_CANCEL, GTK_RESPONSE_REJECT, NULL);
1462
1463 frame = gtk_frame_new("Hostname / socket name");
Jens Axboef1299092012-03-07 20:00:02 +01001464 /* gtk_dialog_get_content_area() is 2.14 and newer */
1465 vbox = GTK_DIALOG(dialog)->vbox;
Jens Axboea7a42ce2012-03-02 13:12:04 +01001466 gtk_box_pack_start(GTK_BOX(vbox), frame, FALSE, FALSE, 5);
1467
1468 box = gtk_vbox_new(FALSE, 6);
1469 gtk_container_add(GTK_CONTAINER(frame), box);
1470
1471 hbox = gtk_hbox_new(TRUE, 10);
1472 gtk_box_pack_start(GTK_BOX(box), hbox, FALSE, FALSE, 0);
Jens Axboe62bc9372012-03-07 11:45:07 +01001473 cw.hentry = gtk_entry_new();
1474 gtk_entry_set_text(GTK_ENTRY(cw.hentry), "localhost");
1475 gtk_box_pack_start(GTK_BOX(hbox), cw.hentry, TRUE, TRUE, 0);
Jens Axboea7a42ce2012-03-02 13:12:04 +01001476
1477 frame = gtk_frame_new("Port");
1478 gtk_box_pack_start(GTK_BOX(vbox), frame, FALSE, FALSE, 5);
1479 box = gtk_vbox_new(FALSE, 10);
1480 gtk_container_add(GTK_CONTAINER(frame), box);
1481
1482 hbox = gtk_hbox_new(TRUE, 4);
1483 gtk_box_pack_start(GTK_BOX(box), hbox, FALSE, FALSE, 0);
1484 pentry = create_spinbutton(hbox, 1, 65535, FIO_NET_PORT);
1485
1486 frame = gtk_frame_new("Type");
1487 gtk_box_pack_start(GTK_BOX(vbox), frame, FALSE, FALSE, 5);
1488 box = gtk_vbox_new(FALSE, 10);
1489 gtk_container_add(GTK_CONTAINER(frame), box);
1490
1491 hbox = gtk_hbox_new(TRUE, 4);
1492 gtk_box_pack_start(GTK_BOX(box), hbox, FALSE, FALSE, 0);
1493
Jens Axboe62bc9372012-03-07 11:45:07 +01001494 cw.combo = gtk_combo_box_new_text();
1495 gtk_combo_box_append_text(GTK_COMBO_BOX(cw.combo), "IPv4");
1496 gtk_combo_box_append_text(GTK_COMBO_BOX(cw.combo), "IPv6");
1497 gtk_combo_box_append_text(GTK_COMBO_BOX(cw.combo), "local socket");
1498 gtk_combo_box_set_active(GTK_COMBO_BOX(cw.combo), 0);
Jens Axboea7a42ce2012-03-02 13:12:04 +01001499
Jens Axboe62bc9372012-03-07 11:45:07 +01001500 gtk_container_add(GTK_CONTAINER(hbox), cw.combo);
Jens Axboea7a42ce2012-03-02 13:12:04 +01001501
Jens Axboeb9f3c7e2012-03-02 14:27:17 +01001502 frame = gtk_frame_new("Options");
1503 gtk_box_pack_start(GTK_BOX(vbox), frame, FALSE, FALSE, 5);
1504 box = gtk_vbox_new(FALSE, 10);
1505 gtk_container_add(GTK_CONTAINER(frame), box);
1506
1507 hbox = gtk_hbox_new(TRUE, 4);
1508 gtk_box_pack_start(GTK_BOX(box), hbox, FALSE, FALSE, 0);
1509
Jens Axboe62bc9372012-03-07 11:45:07 +01001510 cw.button = gtk_check_button_new_with_label("Auto-spawn fio backend");
1511 gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(cw.button), 1);
1512 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.");
1513 gtk_box_pack_start(GTK_BOX(hbox), cw.button, FALSE, FALSE, 6);
1514
1515 /*
1516 * Connect edit signal, so we can show/not-show the auto start button
1517 */
1518 g_signal_connect(GTK_OBJECT(cw.hentry), "changed", G_CALLBACK(hostname_cb), &cw);
1519 g_signal_connect(GTK_OBJECT(cw.combo), "changed", G_CALLBACK(hostname_cb), &cw);
Jens Axboeb9f3c7e2012-03-02 14:27:17 +01001520
Jens Axboea7a42ce2012-03-02 13:12:04 +01001521 gtk_widget_show_all(dialog);
1522
1523 if (gtk_dialog_run(GTK_DIALOG(dialog)) != GTK_RESPONSE_ACCEPT) {
1524 gtk_widget_destroy(dialog);
1525 return 1;
1526 }
1527
Jens Axboe62bc9372012-03-07 11:45:07 +01001528 *host = strdup(gtk_entry_get_text(GTK_ENTRY(cw.hentry)));
Jens Axboea7a42ce2012-03-02 13:12:04 +01001529 *port = gtk_spin_button_get_value_as_int(GTK_SPIN_BUTTON(pentry));
1530
Jens Axboe62bc9372012-03-07 11:45:07 +01001531 typeentry = gtk_combo_box_get_active_text(GTK_COMBO_BOX(cw.combo));
Jens Axboea7a42ce2012-03-02 13:12:04 +01001532 if (!typeentry || !strncmp(typeentry, "IPv4", 4))
1533 *type = Fio_client_ipv4;
1534 else if (!strncmp(typeentry, "IPv6", 4))
1535 *type = Fio_client_ipv6;
1536 else
1537 *type = Fio_client_socket;
1538 g_free(typeentry);
1539
Jens Axboe62bc9372012-03-07 11:45:07 +01001540 *server_start = gtk_toggle_button_get_active(GTK_TOGGLE_BUTTON(cw.button));
Jens Axboeb9f3c7e2012-03-02 14:27:17 +01001541
Jens Axboea7a42ce2012-03-02 13:12:04 +01001542 gtk_widget_destroy(dialog);
1543 return 0;
1544}
1545
Jens Axboee0681f32012-03-06 12:14:42 +01001546static void gfio_client_added(struct gui *ui, struct fio_client *client)
1547{
1548 struct gfio_client *gc;
1549
1550 gc = malloc(sizeof(*gc));
1551 memset(gc, 0, sizeof(*gc));
1552 gc->ui = ui;
1553
1554 client->client_data = gc;
1555}
1556
Jens Axboe0420ba62012-02-29 11:16:52 +01001557static void file_open(GtkWidget *w, gpointer data)
1558{
1559 GtkWidget *dialog;
Jens Axboe63a130b2012-03-06 20:08:59 +01001560 struct gui *ui = data;
Jens Axboe0420ba62012-02-29 11:16:52 +01001561 GSList *filenames, *fn_glist;
1562 GtkFileFilter *filter;
Jens Axboea7a42ce2012-03-02 13:12:04 +01001563 char *host;
Jens Axboeb9f3c7e2012-03-02 14:27:17 +01001564 int port, type, server_start;
Jens Axboe0420ba62012-02-29 11:16:52 +01001565
1566 dialog = gtk_file_chooser_dialog_new("Open File",
Jens Axboe63a130b2012-03-06 20:08:59 +01001567 GTK_WINDOW(ui->window),
Jens Axboe0420ba62012-02-29 11:16:52 +01001568 GTK_FILE_CHOOSER_ACTION_OPEN,
1569 GTK_STOCK_CANCEL, GTK_RESPONSE_CANCEL,
1570 GTK_STOCK_OPEN, GTK_RESPONSE_ACCEPT,
1571 NULL);
1572 gtk_file_chooser_set_select_multiple(GTK_FILE_CHOOSER(dialog), TRUE);
1573
1574 filter = gtk_file_filter_new();
1575 gtk_file_filter_add_pattern(filter, "*.fio");
1576 gtk_file_filter_add_pattern(filter, "*.job");
Jens Axboe2d262992012-03-07 08:19:30 +01001577 gtk_file_filter_add_pattern(filter, "*.ini");
Jens Axboe0420ba62012-02-29 11:16:52 +01001578 gtk_file_filter_add_mime_type(filter, "text/fio");
1579 gtk_file_filter_set_name(filter, "Fio job file");
1580 gtk_file_chooser_set_filter(GTK_FILE_CHOOSER(dialog), filter);
1581
1582 if (gtk_dialog_run(GTK_DIALOG(dialog)) != GTK_RESPONSE_ACCEPT) {
1583 gtk_widget_destroy(dialog);
1584 return;
1585 }
1586
1587 fn_glist = gtk_file_chooser_get_filenames(GTK_FILE_CHOOSER(dialog));
Jens Axboea7a42ce2012-03-02 13:12:04 +01001588
1589 gtk_widget_destroy(dialog);
1590
Jens Axboeb9f3c7e2012-03-02 14:27:17 +01001591 if (get_connection_details(&host, &port, &type, &server_start))
Jens Axboea7a42ce2012-03-02 13:12:04 +01001592 goto err;
1593
Jens Axboe0420ba62012-02-29 11:16:52 +01001594 filenames = fn_glist;
1595 while (filenames != NULL) {
Jens Axboee0681f32012-03-06 12:14:42 +01001596 struct fio_client *client;
1597
Jens Axboe63a130b2012-03-06 20:08:59 +01001598 ui->job_files = realloc(ui->job_files, (ui->nr_job_files + 1) * sizeof(char *));
1599 ui->job_files[ui->nr_job_files] = strdup(filenames->data);
1600 ui->nr_job_files++;
Jens Axboe0420ba62012-02-29 11:16:52 +01001601
Jens Axboee0681f32012-03-06 12:14:42 +01001602 client = fio_client_add_explicit(&gfio_client_ops, host, type, port);
1603 if (!client) {
Jens Axboedf06f222012-03-02 13:32:04 +01001604 GError *error;
1605
1606 error = g_error_new(g_quark_from_string("fio"), 1,
1607 "Failed to add client %s", host);
Jens Axboe0420ba62012-02-29 11:16:52 +01001608 report_error(error);
1609 g_error_free(error);
Jens Axboe0420ba62012-02-29 11:16:52 +01001610 }
Jens Axboe63a130b2012-03-06 20:08:59 +01001611 gfio_client_added(ui, client);
Jens Axboe0420ba62012-02-29 11:16:52 +01001612
1613 g_free(filenames->data);
1614 filenames = g_slist_next(filenames);
1615 }
Jens Axboea7a42ce2012-03-02 13:12:04 +01001616 free(host);
Jens Axboe63a130b2012-03-06 20:08:59 +01001617
1618 if (server_start)
1619 gfio_start_server(ui);
Jens Axboea7a42ce2012-03-02 13:12:04 +01001620err:
Jens Axboe0420ba62012-02-29 11:16:52 +01001621 g_slist_free(fn_glist);
Jens Axboe0420ba62012-02-29 11:16:52 +01001622}
1623
1624static void file_save(GtkWidget *w, gpointer data)
1625{
Jens Axboe63a130b2012-03-06 20:08:59 +01001626 struct gui *ui = data;
Jens Axboe0420ba62012-02-29 11:16:52 +01001627 GtkWidget *dialog;
1628
1629 dialog = gtk_file_chooser_dialog_new("Save File",
Jens Axboe63a130b2012-03-06 20:08:59 +01001630 GTK_WINDOW(ui->window),
Jens Axboe0420ba62012-02-29 11:16:52 +01001631 GTK_FILE_CHOOSER_ACTION_SAVE,
1632 GTK_STOCK_CANCEL, GTK_RESPONSE_CANCEL,
1633 GTK_STOCK_SAVE, GTK_RESPONSE_ACCEPT,
1634 NULL);
1635
1636 gtk_file_chooser_set_do_overwrite_confirmation(GTK_FILE_CHOOSER(dialog), TRUE);
1637 gtk_file_chooser_set_current_name(GTK_FILE_CHOOSER(dialog), "Untitled document");
1638
1639 if (gtk_dialog_run(GTK_DIALOG(dialog)) == GTK_RESPONSE_ACCEPT) {
1640 char *filename;
1641
1642 filename = gtk_file_chooser_get_filename(GTK_FILE_CHOOSER(dialog));
1643 // save_job_file(filename);
1644 g_free(filename);
1645 }
1646 gtk_widget_destroy(dialog);
1647}
1648
Jens Axboe9b260bd2012-03-06 11:02:52 +01001649static void view_log_destroy(GtkWidget *w, gpointer data)
1650{
1651 struct gui *ui = (struct gui *) data;
1652
1653 gtk_widget_ref(ui->log_tree);
1654 gtk_container_remove(GTK_CONTAINER(w), ui->log_tree);
1655 gtk_widget_destroy(w);
Jens Axboe4cbe7212012-03-06 13:36:17 +01001656 ui->log_view = NULL;
Jens Axboe9b260bd2012-03-06 11:02:52 +01001657}
1658
1659static void view_log(GtkWidget *w, gpointer data)
1660{
Jens Axboe4cbe7212012-03-06 13:36:17 +01001661 GtkWidget *win, *scroll, *vbox, *box;
1662 struct gui *ui = (struct gui *) data;
Jens Axboe9b260bd2012-03-06 11:02:52 +01001663
Jens Axboe4cbe7212012-03-06 13:36:17 +01001664 if (ui->log_view)
1665 return;
1666
1667 ui->log_view = win = gtk_window_new(GTK_WINDOW_TOPLEVEL);
Jens Axboe9b260bd2012-03-06 11:02:52 +01001668 gtk_window_set_title(GTK_WINDOW(win), "Log");
Jens Axboe4cbe7212012-03-06 13:36:17 +01001669 gtk_window_set_default_size(GTK_WINDOW(win), 700, 500);
Jens Axboe9b260bd2012-03-06 11:02:52 +01001670
Jens Axboe4cbe7212012-03-06 13:36:17 +01001671 scroll = gtk_scrolled_window_new(NULL, NULL);
Jens Axboe9b260bd2012-03-06 11:02:52 +01001672
Jens Axboe4cbe7212012-03-06 13:36:17 +01001673 gtk_container_set_border_width(GTK_CONTAINER(scroll), 5);
1674
1675 gtk_scrolled_window_set_policy(GTK_SCROLLED_WINDOW(scroll), GTK_POLICY_AUTOMATIC, GTK_POLICY_AUTOMATIC);
1676
1677 box = gtk_hbox_new(TRUE, 0);
1678 gtk_box_pack_start_defaults(GTK_BOX(box), ui->log_tree);
1679 g_signal_connect(box, "destroy", G_CALLBACK(view_log_destroy), ui);
1680 gtk_scrolled_window_add_with_viewport(GTK_SCROLLED_WINDOW(scroll), box);
1681
1682 vbox = gtk_vbox_new(TRUE, 5);
1683 gtk_box_pack_start_defaults(GTK_BOX(vbox), scroll);
1684
1685 gtk_container_add(GTK_CONTAINER(win), vbox);
Jens Axboe9b260bd2012-03-06 11:02:52 +01001686 gtk_widget_show_all(win);
1687}
1688
Jens Axboe46974a72012-03-02 19:34:13 +01001689static void preferences(GtkWidget *w, gpointer data)
1690{
Jens Axboef3e84402012-03-07 13:14:32 +01001691 GtkWidget *dialog, *frame, *box, **buttons, *vbox, *font;
Jens Axboe46974a72012-03-02 19:34:13 +01001692 int i;
1693
1694 dialog = gtk_dialog_new_with_buttons("Preferences",
1695 GTK_WINDOW(ui.window),
1696 GTK_DIALOG_DESTROY_WITH_PARENT,
1697 GTK_STOCK_OK, GTK_RESPONSE_ACCEPT,
1698 GTK_STOCK_CANCEL, GTK_RESPONSE_REJECT,
1699 NULL);
1700
Jens Axboe0b8d11e2012-03-02 19:44:15 +01001701 frame = gtk_frame_new("Debug logging");
Jens Axboe46974a72012-03-02 19:34:13 +01001702 gtk_box_pack_start(GTK_BOX(GTK_DIALOG(dialog)->vbox), frame, FALSE, FALSE, 5);
Jens Axboef3e84402012-03-07 13:14:32 +01001703
1704 vbox = gtk_vbox_new(FALSE, 6);
1705 gtk_container_add(GTK_CONTAINER(frame), vbox);
1706
Jens Axboe46974a72012-03-02 19:34:13 +01001707 box = gtk_hbox_new(FALSE, 6);
Jens Axboef3e84402012-03-07 13:14:32 +01001708 gtk_container_add(GTK_CONTAINER(vbox), box);
Jens Axboe46974a72012-03-02 19:34:13 +01001709
1710 buttons = malloc(sizeof(GtkWidget *) * FD_DEBUG_MAX);
1711
1712 for (i = 0; i < FD_DEBUG_MAX; i++) {
Jens Axboef3e84402012-03-07 13:14:32 +01001713 if (i == 7) {
1714 box = gtk_hbox_new(FALSE, 6);
1715 gtk_container_add(GTK_CONTAINER(vbox), box);
1716 }
1717
1718
Jens Axboe46974a72012-03-02 19:34:13 +01001719 buttons[i] = gtk_check_button_new_with_label(debug_levels[i].name);
Jens Axboe0b8d11e2012-03-02 19:44:15 +01001720 gtk_widget_set_tooltip_text(buttons[i], debug_levels[i].help);
Jens Axboe46974a72012-03-02 19:34:13 +01001721 gtk_box_pack_start(GTK_BOX(box), buttons[i], FALSE, FALSE, 6);
1722 }
1723
Jens Axboef3e84402012-03-07 13:14:32 +01001724 frame = gtk_frame_new("Graph font");
1725 gtk_box_pack_start(GTK_BOX(GTK_DIALOG(dialog)->vbox), frame, FALSE, FALSE, 5);
1726 vbox = gtk_vbox_new(FALSE, 6);
1727 gtk_container_add(GTK_CONTAINER(frame), vbox);
1728
1729 font = gtk_font_button_new();
1730 gtk_box_pack_start(GTK_BOX(vbox), font, FALSE, FALSE, 5);
1731
Jens Axboe46974a72012-03-02 19:34:13 +01001732 gtk_widget_show_all(dialog);
1733
1734 if (gtk_dialog_run(GTK_DIALOG(dialog)) != GTK_RESPONSE_ACCEPT) {
1735 gtk_widget_destroy(dialog);
1736 return;
1737 }
1738
1739 for (i = 0; i < FD_DEBUG_MAX; i++) {
1740 int set;
1741
1742 set = gtk_toggle_button_get_active(GTK_TOGGLE_BUTTON(buttons[i]));
1743 if (set)
1744 fio_debug |= (1UL << i);
1745 }
1746
Jens Axboef3e84402012-03-07 13:14:32 +01001747 gfio_graph_font = strdup(gtk_font_button_get_font_name(GTK_FONT_BUTTON(font)));
Jens Axboe46974a72012-03-02 19:34:13 +01001748 gtk_widget_destroy(dialog);
1749}
1750
Jens Axboe0420ba62012-02-29 11:16:52 +01001751static void about_dialog(GtkWidget *w, gpointer data)
1752{
Jens Axboe81e4ea62012-03-07 14:18:28 +01001753 const char *authors[] = {
1754 "Jens Axboe <axboe@kernel.dk>",
1755 "Stephen Carmeron <stephenmcameron@gmail.com>",
1756 NULL
1757 };
Jens Axboe84a72ed2012-03-07 14:24:57 +01001758 const char *license[] = {
1759 "Fio is free software; you can redistribute it and/or modify "
1760 "it under the terms of the GNU General Public License as published by "
1761 "the Free Software Foundation; either version 2 of the License, or "
1762 "(at your option) any later version.\n",
1763 "Fio is distributed in the hope that it will be useful, "
1764 "but WITHOUT ANY WARRANTY; without even the implied warranty of "
1765 "MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the "
1766 "GNU General Public License for more details.\n",
1767 "You should have received a copy of the GNU General Public License "
1768 "along with Fio; if not, write to the Free Software Foundation, Inc., "
1769 "51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA\n"
1770 };
1771 char *license_trans;
1772
1773 license_trans = g_strconcat(license[0], "\n", license[1], "\n",
1774 license[2], "\n", NULL);
Jens Axboe81e4ea62012-03-07 14:18:28 +01001775
Jens Axboe0420ba62012-02-29 11:16:52 +01001776 gtk_show_about_dialog(NULL,
1777 "program-name", "gfio",
1778 "comments", "Gtk2 UI for fio",
Jens Axboe84a72ed2012-03-07 14:24:57 +01001779 "license", license_trans,
Jens Axboe81e4ea62012-03-07 14:18:28 +01001780 "website", "http://git.kernel.dk/?p=fio.git;a=summary",
1781 "authors", authors,
Jens Axboe0420ba62012-02-29 11:16:52 +01001782 "version", fio_version_string,
Jens Axboe81e4ea62012-03-07 14:18:28 +01001783 "copyright", "© 2012 Jens Axboe <axboe@kernel.dk>",
Jens Axboe0420ba62012-02-29 11:16:52 +01001784 "logo-icon-name", "fio",
1785 /* Must be last: */
Jens Axboe81e4ea62012-03-07 14:18:28 +01001786 "wrap-license", TRUE,
Jens Axboe0420ba62012-02-29 11:16:52 +01001787 NULL);
Jens Axboe84a72ed2012-03-07 14:24:57 +01001788
1789 g_free (license_trans);
Jens Axboe0420ba62012-02-29 11:16:52 +01001790}
1791
1792static GtkActionEntry menu_items[] = {
Jens Axboe46974a72012-03-02 19:34:13 +01001793 { "FileMenuAction", GTK_STOCK_FILE, "File", NULL, NULL, NULL},
Jens Axboe9b260bd2012-03-06 11:02:52 +01001794 { "ViewMenuAction", GTK_STOCK_FILE, "View", NULL, NULL, NULL},
Jens Axboe46974a72012-03-02 19:34:13 +01001795 { "HelpMenuAction", GTK_STOCK_HELP, "Help", NULL, NULL, NULL},
1796 { "OpenFile", GTK_STOCK_OPEN, NULL, "<Control>O", NULL, G_CALLBACK(file_open) },
1797 { "SaveFile", GTK_STOCK_SAVE, NULL, "<Control>S", NULL, G_CALLBACK(file_save) },
1798 { "Preferences", GTK_STOCK_PREFERENCES, NULL, "<Control>p", NULL, G_CALLBACK(preferences) },
Jens Axboe9b260bd2012-03-06 11:02:52 +01001799 { "ViewLog", NULL, "Log", "<Control>l", NULL, G_CALLBACK(view_log) },
Jens Axboe46974a72012-03-02 19:34:13 +01001800 { "Quit", GTK_STOCK_QUIT, NULL, "<Control>Q", NULL, G_CALLBACK(quit_clicked) },
1801 { "About", GTK_STOCK_ABOUT, NULL, NULL, NULL, G_CALLBACK(about_dialog) },
Jens Axboe0420ba62012-02-29 11:16:52 +01001802};
Jens Axboe3e47bd22012-02-29 13:45:02 +01001803static gint nmenu_items = sizeof(menu_items) / sizeof(menu_items[0]);
Jens Axboe0420ba62012-02-29 11:16:52 +01001804
1805static const gchar *ui_string = " \
1806 <ui> \
1807 <menubar name=\"MainMenu\"> \
1808 <menu name=\"FileMenu\" action=\"FileMenuAction\"> \
1809 <menuitem name=\"Open\" action=\"OpenFile\" /> \
1810 <menuitem name=\"Save\" action=\"SaveFile\" /> \
1811 <separator name=\"Separator\"/> \
Jens Axboe46974a72012-03-02 19:34:13 +01001812 <menuitem name=\"Preferences\" action=\"Preferences\" /> \
1813 <separator name=\"Separator2\"/> \
Jens Axboe0420ba62012-02-29 11:16:52 +01001814 <menuitem name=\"Quit\" action=\"Quit\" /> \
1815 </menu> \
Jens Axboe9b260bd2012-03-06 11:02:52 +01001816 <menu name=\"ViewMenu\" action=\"ViewMenuAction\"> \
1817 <menuitem name=\"Log\" action=\"ViewLog\" /> \
1818 </menu>\
Jens Axboe0420ba62012-02-29 11:16:52 +01001819 <menu name=\"Help\" action=\"HelpMenuAction\"> \
1820 <menuitem name=\"About\" action=\"About\" /> \
1821 </menu> \
1822 </menubar> \
1823 </ui> \
1824";
1825
Jens Axboe4cbe7212012-03-06 13:36:17 +01001826static GtkWidget *get_menubar_menu(GtkWidget *window, GtkUIManager *ui_manager,
1827 struct gui *ui)
Jens Axboe0420ba62012-02-29 11:16:52 +01001828{
1829 GtkActionGroup *action_group = gtk_action_group_new("Menu");
1830 GError *error = 0;
1831
1832 action_group = gtk_action_group_new("Menu");
Jens Axboe4cbe7212012-03-06 13:36:17 +01001833 gtk_action_group_add_actions(action_group, menu_items, nmenu_items, ui);
Jens Axboe0420ba62012-02-29 11:16:52 +01001834
1835 gtk_ui_manager_insert_action_group(ui_manager, action_group, 0);
1836 gtk_ui_manager_add_ui_from_string(GTK_UI_MANAGER(ui_manager), ui_string, -1, &error);
1837
1838 gtk_window_add_accel_group(GTK_WINDOW(window), gtk_ui_manager_get_accel_group(ui_manager));
1839 return gtk_ui_manager_get_widget(ui_manager, "/MainMenu");
1840}
1841
1842void gfio_ui_setup(GtkSettings *settings, GtkWidget *menubar,
1843 GtkWidget *vbox, GtkUIManager *ui_manager)
1844{
1845 gtk_box_pack_start(GTK_BOX(vbox), menubar, FALSE, FALSE, 0);
1846}
1847
Stephen M. Cameronff1f3282012-02-24 08:17:30 +01001848static void init_ui(int *argc, char **argv[], struct gui *ui)
1849{
Jens Axboe0420ba62012-02-29 11:16:52 +01001850 GtkSettings *settings;
1851 GtkUIManager *uimanager;
Jens Axboe843ad232012-02-29 11:44:53 +01001852 GtkWidget *menu, *probe, *probe_frame, *probe_box;
Stephen M. Cameronaaa71f62012-03-07 14:47:03 +01001853 GdkColor white;
Jens Axboe0420ba62012-02-29 11:16:52 +01001854
1855 memset(ui, 0, sizeof(*ui));
Stephen M. Cameron45032dd2012-02-24 08:17:31 +01001856
Stephen M. Cameron2839f0c2012-02-24 08:17:31 +01001857 /* Magical g*thread incantation, you just need this thread stuff.
Stephen M. Cameron04cc6b72012-02-24 08:17:31 +01001858 * Without it, the update that happens in gfio_update_thread_status
Stephen M. Cameron2839f0c2012-02-24 08:17:31 +01001859 * doesn't really happen in a timely fashion, you need expose events
1860 */
Jens Axboeed727a42012-03-02 12:14:40 +01001861 if (!g_thread_supported())
Stephen M. Cameron2839f0c2012-02-24 08:17:31 +01001862 g_thread_init(NULL);
1863 gdk_threads_init();
1864
Stephen M. Cameronff1f3282012-02-24 08:17:30 +01001865 gtk_init(argc, argv);
Jens Axboe0420ba62012-02-29 11:16:52 +01001866 settings = gtk_settings_get_default();
1867 gtk_settings_set_long_property(settings, "gtk_tooltip_timeout", 10, "gfio setting");
1868 g_type_init();
Stephen M. Cameronff1f3282012-02-24 08:17:30 +01001869
1870 ui->window = gtk_window_new(GTK_WINDOW_TOPLEVEL);
1871 gtk_window_set_title(GTK_WINDOW(ui->window), "fio");
Jens Axboef3e84402012-03-07 13:14:32 +01001872 gtk_window_set_default_size(GTK_WINDOW(ui->window), 700, 700);
Stephen M. Cameronff1f3282012-02-24 08:17:30 +01001873
Jens Axboe0420ba62012-02-29 11:16:52 +01001874 g_signal_connect(ui->window, "delete-event", G_CALLBACK(quit_clicked), NULL);
1875 g_signal_connect(ui->window, "destroy", G_CALLBACK(quit_clicked), NULL);
Stephen M. Cameronff1f3282012-02-24 08:17:30 +01001876
Stephen M. Cameron5b7573a2012-02-24 08:17:31 +01001877 ui->vbox = gtk_vbox_new(FALSE, 0);
1878 gtk_container_add(GTK_CONTAINER (ui->window), ui->vbox);
Stephen M. Cameron04cc6b72012-02-24 08:17:31 +01001879
Jens Axboe0420ba62012-02-29 11:16:52 +01001880 uimanager = gtk_ui_manager_new();
Jens Axboe4cbe7212012-03-06 13:36:17 +01001881 menu = get_menubar_menu(ui->window, uimanager, ui);
Jens Axboe0420ba62012-02-29 11:16:52 +01001882 gfio_ui_setup(settings, menu, ui->vbox, uimanager);
1883
Stephen M. Cameron04cc6b72012-02-24 08:17:31 +01001884 /*
Stephen M. Cameronc36f98d2012-02-24 08:17:32 +01001885 * Set up alignments for widgets at the top of ui,
1886 * align top left, expand horizontally but not vertically
1887 */
1888 ui->topalign = gtk_alignment_new(0, 0, 1, 0);
Jens Axboe3ec62ec2012-03-01 12:01:29 +01001889 ui->topvbox = gtk_vbox_new(FALSE, 3);
Stephen M. Cameronc36f98d2012-02-24 08:17:32 +01001890 gtk_container_add(GTK_CONTAINER(ui->topalign), ui->topvbox);
Stephen M. Camerone1645342012-02-24 08:17:32 +01001891 gtk_box_pack_start(GTK_BOX(ui->vbox), ui->topalign, FALSE, FALSE, 0);
Stephen M. Cameronc36f98d2012-02-24 08:17:32 +01001892
Jens Axboe3e47bd22012-02-29 13:45:02 +01001893 probe = gtk_frame_new("Job");
Jens Axboe843ad232012-02-29 11:44:53 +01001894 gtk_box_pack_start(GTK_BOX(ui->topvbox), probe, TRUE, FALSE, 3);
1895 probe_frame = gtk_vbox_new(FALSE, 3);
1896 gtk_container_add(GTK_CONTAINER(probe), probe_frame);
1897
1898 probe_box = gtk_hbox_new(FALSE, 3);
1899 gtk_box_pack_start(GTK_BOX(probe_frame), probe_box, TRUE, FALSE, 3);
Jens Axboe843ad232012-02-29 11:44:53 +01001900 ui->probe.hostname = new_info_label_in_frame(probe_box, "Host");
1901 ui->probe.os = new_info_label_in_frame(probe_box, "OS");
1902 ui->probe.arch = new_info_label_in_frame(probe_box, "Architecture");
1903 ui->probe.fio_ver = new_info_label_in_frame(probe_box, "Fio version");
1904
Jens Axboe3e47bd22012-02-29 13:45:02 +01001905 probe_box = gtk_hbox_new(FALSE, 3);
1906 gtk_box_pack_start(GTK_BOX(probe_frame), probe_box, TRUE, FALSE, 3);
Jens Axboe807f9972012-03-02 10:25:24 +01001907
Jens Axboeca850992012-03-05 20:04:43 +01001908 ui->eta.name = new_info_entry_in_frame(probe_box, "Name");
1909 ui->eta.iotype = new_info_entry_in_frame(probe_box, "IO");
1910 ui->eta.ioengine = new_info_entry_in_frame(probe_box, "IO Engine");
1911 ui->eta.iodepth = new_info_entry_in_frame(probe_box, "IO Depth");
1912 ui->eta.jobs = new_info_entry_in_frame(probe_box, "Jobs");
1913 ui->eta.files = new_info_entry_in_frame(probe_box, "Open files");
Jens Axboe3e47bd22012-02-29 13:45:02 +01001914
1915 probe_box = gtk_hbox_new(FALSE, 3);
1916 gtk_box_pack_start(GTK_BOX(probe_frame), probe_box, TRUE, FALSE, 3);
Jens Axboeca850992012-03-05 20:04:43 +01001917 ui->eta.read_bw = new_info_entry_in_frame(probe_box, "Read BW");
1918 ui->eta.read_iops = new_info_entry_in_frame(probe_box, "IOPS");
1919 ui->eta.write_bw = new_info_entry_in_frame(probe_box, "Write BW");
1920 ui->eta.write_iops = new_info_entry_in_frame(probe_box, "IOPS");
Jens Axboe807f9972012-03-02 10:25:24 +01001921
1922 /*
1923 * Only add this if we have a commit rate
1924 */
1925#if 0
1926 probe_box = gtk_hbox_new(FALSE, 3);
1927 gtk_box_pack_start(GTK_BOX(probe_frame), probe_box, TRUE, FALSE, 3);
1928
Jens Axboe3e47bd22012-02-29 13:45:02 +01001929 ui->eta.cr_bw = new_info_label_in_frame(probe_box, "Commit BW");
1930 ui->eta.cr_iops = new_info_label_in_frame(probe_box, "Commit IOPS");
1931
Jens Axboe3e47bd22012-02-29 13:45:02 +01001932 ui->eta.cw_bw = new_info_label_in_frame(probe_box, "Commit BW");
1933 ui->eta.cw_iops = new_info_label_in_frame(probe_box, "Commit IOPS");
Jens Axboe807f9972012-03-02 10:25:24 +01001934#endif
Jens Axboe3e47bd22012-02-29 13:45:02 +01001935
Stephen M. Cameron45032dd2012-02-24 08:17:31 +01001936 /*
Jens Axboe2fd3bb02012-03-07 08:07:39 +01001937 * Set up a drawing area and IOPS and bandwidth graphs
Stephen M. Cameron736f2df2012-02-24 08:17:32 +01001938 */
Stephen M. Cameronaaa71f62012-03-07 14:47:03 +01001939 gdk_color_parse("white", &white);
Jens Axboe2fd3bb02012-03-07 08:07:39 +01001940 ui->drawing_area = gtk_drawing_area_new();
Stephen M. Cameron3ea48b82012-03-07 19:40:58 +01001941 ui->drawing_area_xdim = DRAWING_AREA_XDIM;
1942 ui->drawing_area_ydim = DRAWING_AREA_YDIM;
Jens Axboe2fd3bb02012-03-07 08:07:39 +01001943 gtk_widget_set_size_request(GTK_WIDGET(ui->drawing_area),
Stephen M. Cameron3ea48b82012-03-07 19:40:58 +01001944 ui->drawing_area_xdim, ui->drawing_area_ydim);
Stephen M. Cameronaaa71f62012-03-07 14:47:03 +01001945 gtk_widget_modify_bg(ui->drawing_area, GTK_STATE_NORMAL, &white);
Jens Axboe2fd3bb02012-03-07 08:07:39 +01001946 g_signal_connect(G_OBJECT(ui->drawing_area), "expose_event",
1947 G_CALLBACK (on_expose_drawing_area), ui);
Stephen M. Cameron3ea48b82012-03-07 19:40:58 +01001948 g_signal_connect(G_OBJECT(ui->drawing_area), "configure_event",
1949 G_CALLBACK (on_config_drawing_area), ui);
Stephen M. Cameron736f2df2012-02-24 08:17:32 +01001950 ui->scrolled_window = gtk_scrolled_window_new(NULL, NULL);
1951 gtk_scrolled_window_set_policy(GTK_SCROLLED_WINDOW(ui->scrolled_window),
1952 GTK_POLICY_AUTOMATIC, GTK_POLICY_AUTOMATIC);
Jens Axboe2fd3bb02012-03-07 08:07:39 +01001953 gtk_scrolled_window_add_with_viewport(GTK_SCROLLED_WINDOW(ui->scrolled_window),
1954 ui->drawing_area);
Stephen M. Camerone1645342012-02-24 08:17:32 +01001955 gtk_box_pack_start(GTK_BOX(ui->vbox), ui->scrolled_window,
1956 TRUE, TRUE, 0);
Stephen M. Cameron736f2df2012-02-24 08:17:32 +01001957
Jens Axboe2fd3bb02012-03-07 08:07:39 +01001958 setup_iops_graph(ui);
1959 setup_bandwidth_graph(ui);
1960
Stephen M. Cameronc36f98d2012-02-24 08:17:32 +01001961 /*
1962 * Set up alignments for widgets at the bottom of ui,
1963 * align bottom left, expand horizontally but not vertically
1964 */
1965 ui->bottomalign = gtk_alignment_new(0, 1, 1, 0);
1966 ui->buttonbox = gtk_hbox_new(FALSE, 0);
1967 gtk_container_add(GTK_CONTAINER(ui->bottomalign), ui->buttonbox);
Stephen M. Camerone1645342012-02-24 08:17:32 +01001968 gtk_box_pack_start(GTK_BOX(ui->vbox), ui->bottomalign,
1969 FALSE, FALSE, 0);
Stephen M. Cameronc36f98d2012-02-24 08:17:32 +01001970
Stephen M. Cameronf3074002012-02-24 08:17:30 +01001971 add_buttons(ui, buttonspeclist, ARRAYSIZE(buttonspeclist));
Jens Axboe3ec62ec2012-03-01 12:01:29 +01001972
1973 /*
1974 * Set up thread status progress bar
1975 */
1976 ui->thread_status_pb = gtk_progress_bar_new();
1977 gtk_progress_bar_set_fraction(GTK_PROGRESS_BAR(ui->thread_status_pb), 0.0);
Jens Axboe8663ea62012-03-02 14:04:30 +01001978 gtk_progress_bar_set_text(GTK_PROGRESS_BAR(ui->thread_status_pb), "No connections");
Jens Axboe3ec62ec2012-03-01 12:01:29 +01001979 gtk_container_add(GTK_CONTAINER(ui->buttonbox), ui->thread_status_pb);
1980
Jens Axboe9b260bd2012-03-06 11:02:52 +01001981 gfio_ui_setup_log(ui);
Jens Axboe3ec62ec2012-03-01 12:01:29 +01001982
Stephen M. Cameronff1f3282012-02-24 08:17:30 +01001983 gtk_widget_show_all(ui->window);
1984}
1985
Stephen M. Cameron8232e282012-02-24 08:17:31 +01001986int main(int argc, char *argv[], char *envp[])
Stephen M. Cameronff1f3282012-02-24 08:17:30 +01001987{
Stephen M. Cameron8232e282012-02-24 08:17:31 +01001988 if (initialize_fio(envp))
1989 return 1;
Jens Axboe0420ba62012-02-29 11:16:52 +01001990 if (fio_init_options())
1991 return 1;
Stephen M. Camerona1820202012-02-24 08:17:31 +01001992
Stephen M. Cameronff1f3282012-02-24 08:17:30 +01001993 init_ui(&argc, &argv, &ui);
Stephen M. Cameron5b7573a2012-02-24 08:17:31 +01001994
Stephen M. Cameron2839f0c2012-02-24 08:17:31 +01001995 gdk_threads_enter();
Stephen M. Cameronff1f3282012-02-24 08:17:30 +01001996 gtk_main();
Stephen M. Cameron2839f0c2012-02-24 08:17:31 +01001997 gdk_threads_leave();
Stephen M. Cameronff1f3282012-02-24 08:17:30 +01001998 return 0;
1999}