blob: 5d0b866708559c1adb5253306c83fe4fc50e3fd8 [file] [log] [blame]
Jens Axboe79d16312010-03-04 12:43:20 +01001#include "fio.h"
2#include "profile.h"
3#include "debug.h"
4#include "flist.h"
Jens Axboee2de69d2010-03-04 14:05:48 +01005#include "options.h"
Jens Axboe79d16312010-03-04 12:43:20 +01006
7static FLIST_HEAD(profile_list);
8
Jens Axboe15dc1932010-03-05 10:59:06 +01009struct profile_ops *find_profile(const char *profile)
Jens Axboe79d16312010-03-04 12:43:20 +010010{
Jens Axboe15dc1932010-03-05 10:59:06 +010011 struct profile_ops *ops = NULL;
Jens Axboe79d16312010-03-04 12:43:20 +010012 struct flist_head *n;
13
Jens Axboe79d16312010-03-04 12:43:20 +010014 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 Axboe15dc1932010-03-05 10:59:06 +010022 return ops;
23}
24
25int 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 Axboe79d16312010-03-04 12:43:20 +010032 if (ops) {
Jens Axboed4afedf2013-05-22 22:21:29 +020033 if (ops->prep_cmd()) {
Jens Axboe90777552013-05-24 09:09:39 +020034 log_err("fio: profile %s prep failed\n", profile);
Jens Axboed4afedf2013-05-22 22:21:29 +020035 return 1;
36 }
Jens Axboe46bcd492012-03-14 11:31:21 +010037 add_job_opts(ops->cmdline, FIO_CLIENT_TYPE_CLI);
Jens Axboe79d16312010-03-04 12:43:20 +010038 return 0;
39 }
40
41 log_err("fio: profile '%s' not found\n", profile);
42 return 1;
43}
44
Jens Axboe07b32322010-03-05 09:48:44 +010045static int add_profile_options(struct profile_ops *ops)
Jens Axboee2de69d2010-03-04 14:05:48 +010046{
Jens Axboe07b32322010-03-05 09:48:44 +010047 struct fio_option *o;
Jens Axboe3c3ed072012-03-27 09:12:39 +020048
Jens Axboee2de69d2010-03-04 14:05:48 +010049 if (!ops->options)
Jens Axboe07b32322010-03-05 09:48:44 +010050 return 0;
Jens Axboee2de69d2010-03-04 14:05:48 +010051
Jens Axboe07b32322010-03-05 09:48:44 +010052 o = ops->options;
53 while (o->name) {
54 o->prof_name = ops->name;
55 if (add_option(o))
56 return 1;
57 o++;
Jens Axboee2de69d2010-03-04 14:05:48 +010058 }
Jens Axboe07b32322010-03-05 09:48:44 +010059
60 return 0;
Jens Axboee2de69d2010-03-04 14:05:48 +010061}
62
Jens Axboe07b32322010-03-05 09:48:44 +010063int register_profile(struct profile_ops *ops)
Jens Axboe79d16312010-03-04 12:43:20 +010064{
Jens Axboe07b32322010-03-05 09:48:44 +010065 int ret;
66
Jens Axboe79d16312010-03-04 12:43:20 +010067 dprint(FD_PROFILE, "register profile '%s'\n", ops->name);
Jens Axboe07b32322010-03-05 09:48:44 +010068
Jens Axboef5b6bb82010-03-05 10:09:59 +010069 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 Axboe07b32322010-03-05 09:48:44 +010077 return ret;
Jens Axboe79d16312010-03-04 12:43:20 +010078}
79
80void unregister_profile(struct profile_ops *ops)
81{
82 dprint(FD_PROFILE, "unregister profile '%s'\n", ops->name);
83 flist_del(&ops->list);
Jens Axboe07b32322010-03-05 09:48:44 +010084 invalidate_profile_options(ops->name);
Jens Axboef5b6bb82010-03-05 10:09:59 +010085 del_opt_posval("profile", ops->name);
Jens Axboe79d16312010-03-04 12:43:20 +010086}
Jens Axboe15dc1932010-03-05 10:59:06 +010087
88void 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 Axboed72be542012-11-30 19:37:46 +010099 if (ops->io_ops) {
Jens Axboe7eb36572010-03-08 13:58:49 +0100100 td->prof_io_ops = *ops->io_ops;
Jens Axboed72be542012-11-30 19:37:46 +0100101 td->flags |= TD_F_PROFILE_OPS;
102 }
Jens Axboe15dc1932010-03-05 10:59:06 +0100103}
Jens Axboe58c55ba2010-03-09 12:20:08 +0100104
105int 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
115void 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}