Jens Axboe | 79d1631 | 2010-03-04 12:43:20 +0100 | [diff] [blame] | 1 | #include "fio.h" |
| 2 | #include "profile.h" |
| 3 | #include "debug.h" |
| 4 | #include "flist.h" |
Jens Axboe | e2de69d | 2010-03-04 14:05:48 +0100 | [diff] [blame] | 5 | #include "options.h" |
Jens Axboe | 79d1631 | 2010-03-04 12:43:20 +0100 | [diff] [blame] | 6 | |
| 7 | static FLIST_HEAD(profile_list); |
| 8 | |
Jens Axboe | 15dc193 | 2010-03-05 10:59:06 +0100 | [diff] [blame] | 9 | struct profile_ops *find_profile(const char *profile) |
Jens Axboe | 79d1631 | 2010-03-04 12:43:20 +0100 | [diff] [blame] | 10 | { |
Jens Axboe | 15dc193 | 2010-03-05 10:59:06 +0100 | [diff] [blame] | 11 | struct profile_ops *ops = NULL; |
Jens Axboe | 79d1631 | 2010-03-04 12:43:20 +0100 | [diff] [blame] | 12 | struct flist_head *n; |
| 13 | |
Jens Axboe | 79d1631 | 2010-03-04 12:43:20 +0100 | [diff] [blame] | 14 | flist_for_each(n, &profile_list) { |
| 15 | ops = flist_entry(n, struct profile_ops, list); |
| 16 | if (!strcmp(profile, ops->name)) |
| 17 | break; |
| 18 | |
| 19 | ops = NULL; |
| 20 | } |
| 21 | |
Jens Axboe | 15dc193 | 2010-03-05 10:59:06 +0100 | [diff] [blame] | 22 | return ops; |
| 23 | } |
| 24 | |
| 25 | int load_profile(const char *profile) |
| 26 | { |
| 27 | struct profile_ops *ops; |
| 28 | |
| 29 | dprint(FD_PROFILE, "loading profile '%s'\n", profile); |
| 30 | |
| 31 | ops = find_profile(profile); |
Jens Axboe | 79d1631 | 2010-03-04 12:43:20 +0100 | [diff] [blame] | 32 | if (ops) { |
Jens Axboe | d4afedf | 2013-05-22 22:21:29 +0200 | [diff] [blame] | 33 | if (ops->prep_cmd()) { |
Jens Axboe | 9077755 | 2013-05-24 09:09:39 +0200 | [diff] [blame] | 34 | log_err("fio: profile %s prep failed\n", profile); |
Jens Axboe | d4afedf | 2013-05-22 22:21:29 +0200 | [diff] [blame] | 35 | return 1; |
| 36 | } |
Jens Axboe | 46bcd49 | 2012-03-14 11:31:21 +0100 | [diff] [blame] | 37 | add_job_opts(ops->cmdline, FIO_CLIENT_TYPE_CLI); |
Jens Axboe | 79d1631 | 2010-03-04 12:43:20 +0100 | [diff] [blame] | 38 | return 0; |
| 39 | } |
| 40 | |
| 41 | log_err("fio: profile '%s' not found\n", profile); |
| 42 | return 1; |
| 43 | } |
| 44 | |
Jens Axboe | 07b3232 | 2010-03-05 09:48:44 +0100 | [diff] [blame] | 45 | static int add_profile_options(struct profile_ops *ops) |
Jens Axboe | e2de69d | 2010-03-04 14:05:48 +0100 | [diff] [blame] | 46 | { |
Jens Axboe | 07b3232 | 2010-03-05 09:48:44 +0100 | [diff] [blame] | 47 | struct fio_option *o; |
Jens Axboe | 3c3ed07 | 2012-03-27 09:12:39 +0200 | [diff] [blame] | 48 | |
Jens Axboe | e2de69d | 2010-03-04 14:05:48 +0100 | [diff] [blame] | 49 | if (!ops->options) |
Jens Axboe | 07b3232 | 2010-03-05 09:48:44 +0100 | [diff] [blame] | 50 | return 0; |
Jens Axboe | e2de69d | 2010-03-04 14:05:48 +0100 | [diff] [blame] | 51 | |
Jens Axboe | 07b3232 | 2010-03-05 09:48:44 +0100 | [diff] [blame] | 52 | o = ops->options; |
| 53 | while (o->name) { |
| 54 | o->prof_name = ops->name; |
| 55 | if (add_option(o)) |
| 56 | return 1; |
| 57 | o++; |
Jens Axboe | e2de69d | 2010-03-04 14:05:48 +0100 | [diff] [blame] | 58 | } |
Jens Axboe | 07b3232 | 2010-03-05 09:48:44 +0100 | [diff] [blame] | 59 | |
| 60 | return 0; |
Jens Axboe | e2de69d | 2010-03-04 14:05:48 +0100 | [diff] [blame] | 61 | } |
| 62 | |
Jens Axboe | 07b3232 | 2010-03-05 09:48:44 +0100 | [diff] [blame] | 63 | int register_profile(struct profile_ops *ops) |
Jens Axboe | 79d1631 | 2010-03-04 12:43:20 +0100 | [diff] [blame] | 64 | { |
Jens Axboe | 07b3232 | 2010-03-05 09:48:44 +0100 | [diff] [blame] | 65 | int ret; |
| 66 | |
Jens Axboe | 79d1631 | 2010-03-04 12:43:20 +0100 | [diff] [blame] | 67 | dprint(FD_PROFILE, "register profile '%s'\n", ops->name); |
Jens Axboe | 07b3232 | 2010-03-05 09:48:44 +0100 | [diff] [blame] | 68 | |
Jens Axboe | f5b6bb8 | 2010-03-05 10:09:59 +0100 | [diff] [blame] | 69 | ret = add_profile_options(ops); |
| 70 | if (!ret) { |
| 71 | flist_add_tail(&ops->list, &profile_list); |
| 72 | add_opt_posval("profile", ops->name, ops->desc); |
| 73 | return 0; |
| 74 | } |
| 75 | |
| 76 | invalidate_profile_options(ops->name); |
Jens Axboe | 07b3232 | 2010-03-05 09:48:44 +0100 | [diff] [blame] | 77 | return ret; |
Jens Axboe | 79d1631 | 2010-03-04 12:43:20 +0100 | [diff] [blame] | 78 | } |
| 79 | |
| 80 | void unregister_profile(struct profile_ops *ops) |
| 81 | { |
| 82 | dprint(FD_PROFILE, "unregister profile '%s'\n", ops->name); |
| 83 | flist_del(&ops->list); |
Jens Axboe | 07b3232 | 2010-03-05 09:48:44 +0100 | [diff] [blame] | 84 | invalidate_profile_options(ops->name); |
Jens Axboe | f5b6bb8 | 2010-03-05 10:09:59 +0100 | [diff] [blame] | 85 | del_opt_posval("profile", ops->name); |
Jens Axboe | 79d1631 | 2010-03-04 12:43:20 +0100 | [diff] [blame] | 86 | } |
Jens Axboe | 15dc193 | 2010-03-05 10:59:06 +0100 | [diff] [blame] | 87 | |
| 88 | void profile_add_hooks(struct thread_data *td) |
| 89 | { |
| 90 | struct profile_ops *ops; |
| 91 | |
| 92 | if (!exec_profile) |
| 93 | return; |
| 94 | |
| 95 | ops = find_profile(exec_profile); |
| 96 | if (!ops) |
| 97 | return; |
| 98 | |
Jens Axboe | d72be54 | 2012-11-30 19:37:46 +0100 | [diff] [blame] | 99 | if (ops->io_ops) { |
Jens Axboe | 7eb3657 | 2010-03-08 13:58:49 +0100 | [diff] [blame] | 100 | td->prof_io_ops = *ops->io_ops; |
Jens Axboe | d72be54 | 2012-11-30 19:37:46 +0100 | [diff] [blame] | 101 | td->flags |= TD_F_PROFILE_OPS; |
| 102 | } |
Jens Axboe | 15dc193 | 2010-03-05 10:59:06 +0100 | [diff] [blame] | 103 | } |
Jens Axboe | 58c55ba | 2010-03-09 12:20:08 +0100 | [diff] [blame] | 104 | |
| 105 | int profile_td_init(struct thread_data *td) |
| 106 | { |
| 107 | struct prof_io_ops *ops = &td->prof_io_ops; |
| 108 | |
| 109 | if (ops->td_init) |
| 110 | return ops->td_init(td); |
| 111 | |
| 112 | return 0; |
| 113 | } |
| 114 | |
| 115 | void profile_td_exit(struct thread_data *td) |
| 116 | { |
| 117 | struct prof_io_ops *ops = &td->prof_io_ops; |
| 118 | |
| 119 | if (ops->td_exit) |
| 120 | ops->td_exit(td); |
| 121 | } |