01a8cc0f150a5729d612eb76693463cddcc806e0
[vlendec/samba-autobuild/.git] / ctdb / ctdb_bench.c
1 /* 
2    simple ctdb benchmark
3
4    Copyright (C) Andrew Tridgell  2006
5
6    This library is free software; you can redistribute it and/or
7    modify it under the terms of the GNU Lesser General Public
8    License as published by the Free Software Foundation; either
9    version 2 of the License, or (at your option) any later version.
10
11    This library is distributed in the hope that it will be useful,
12    but WITHOUT ANY WARRANTY; without even the implied warranty of
13    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
14    Lesser General Public License for more details.
15
16    You should have received a copy of the GNU Lesser General Public
17    License along with this library; if not, write to the Free Software
18    Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA
19 */
20
21 #include "includes.h"
22 #include "lib/events/events.h"
23 #include "system/filesys.h"
24 #include "popt.h"
25
26 #include <sys/time.h>
27 #include <time.h>
28
29 static struct timeval tp1,tp2;
30
31 static void start_timer(void)
32 {
33         gettimeofday(&tp1,NULL);
34 }
35
36 static double end_timer(void)
37 {
38         gettimeofday(&tp2,NULL);
39         return (tp2.tv_sec + (tp2.tv_usec*1.0e-6)) - 
40                 (tp1.tv_sec + (tp1.tv_usec*1.0e-6));
41 }
42
43
44 static int timelimit = 10;
45 static int num_records = 10;
46 static int num_repeats = 100;
47
48 enum my_functions {FUNC_INCR=1, FUNC_FETCH=2};
49
50 /*
51   ctdb call function to increment an integer
52 */
53 static int incr_func(struct ctdb_call *call)
54 {
55         if (call->record_data.dsize == 0) {
56                 call->new_data = talloc(call, TDB_DATA);
57                 if (call->new_data == NULL) {
58                         return CTDB_ERR_NOMEM;
59                 }
60                 call->new_data->dptr = talloc_size(call, 4);
61                 call->new_data->dsize = 4;
62                 *(uint32_t *)call->new_data->dptr = 0;
63         } else {
64                 call->new_data = &call->record_data;
65         }
66         (*(uint32_t *)call->new_data->dptr)++;
67         return 0;
68 }
69
70 /*
71   ctdb call function to fetch a record
72 */
73 static int fetch_func(struct ctdb_call *call)
74 {
75         call->reply_data = &call->record_data;
76         return 0;
77 }
78
79 /*
80   benchmark incrementing an integer
81 */
82 static void bench_incr(struct ctdb_context *ctdb)
83 {
84         TDB_DATA key, data;
85         int loops=0;
86         int ret, i;
87
88         start_timer();
89
90         while (1) {
91                 uint32_t v = loops % num_records;
92                 key.dptr = &v;
93                 key.dsize = 4;
94                 for (i=0;i<num_repeats;i++) {
95                         ret = ctdb_call(ctdb, key, FUNC_INCR, NULL, NULL);
96                         if (ret != 0) {
97                                 printf("incr call failed - %s\n", ctdb_errstr(ctdb));
98                                 return;
99                         }
100                 }
101                 if (num_repeats * (++loops) % 10000 == 0) {
102                         if (end_timer() > timelimit) break;
103                         printf("Incr: %.2f ops/sec\r", num_repeats*loops/end_timer());
104                         fflush(stdout);
105                 }
106         }
107
108         ret = ctdb_call(ctdb, key, FUNC_FETCH, NULL, &data);
109         if (ret == -1) {
110                 printf("ctdb_call FUNC_FETCH failed - %s\n", ctdb_errstr(ctdb));
111                 return;
112         }
113
114         printf("Incr: %.2f ops/sec (loops=%d val=%d)\n", 
115                num_repeats*loops/end_timer(), loops, *(uint32_t *)data.dptr);
116 }
117
118 /*
119   main program
120 */
121 int main(int argc, const char *argv[])
122 {
123         struct ctdb_context *ctdb;
124         const char *nlist = NULL;
125         const char *transport = "tcp";
126         const char *myaddress = NULL;
127         int self_connect=0;
128
129         struct poptOption popt_options[] = {
130                 POPT_AUTOHELP
131                 { "nlist", 0, POPT_ARG_STRING, &nlist, 0, "node list file", "filename" },
132                 { "listen", 0, POPT_ARG_STRING, &myaddress, 0, "address to listen on", "address" },
133                 { "transport", 0, POPT_ARG_STRING, &transport, 0, "protocol transport", NULL },
134                 { "self-connect", 0, POPT_ARG_NONE, &self_connect, 0, "enable self connect", "boolean" },
135                 { "timelimit", 't', POPT_ARG_INT, &timelimit, 0, "timelimit", "integer" },
136                 { "num-records", 'r', POPT_ARG_INT, &num_records, 0, "num_records", "integer" },
137                 POPT_TABLEEND
138         };
139         int opt;
140         const char **extra_argv;
141         int extra_argc = 0;
142         int ret;
143         poptContext pc;
144         struct event_context *ev;
145
146         pc = poptGetContext(argv[0], argc, argv, popt_options, POPT_CONTEXT_KEEP_FIRST);
147
148         while ((opt = poptGetNextOpt(pc)) != -1) {
149                 switch (opt) {
150                 default:
151                         fprintf(stderr, "Invalid option %s: %s\n", 
152                                 poptBadOption(pc, 0), poptStrerror(opt));
153                         exit(1);
154                 }
155         }
156
157         /* setup the remaining options for the main program to use */
158         extra_argv = poptGetArgs(pc);
159         if (extra_argv) {
160                 extra_argv++;
161                 while (extra_argv[extra_argc]) extra_argc++;
162         }
163
164         if (nlist == NULL || myaddress == NULL) {
165                 printf("You must provide a node list with --nlist and an address with --listen\n");
166                 exit(1);
167         }
168
169         ev = event_context_init(NULL);
170
171         /* initialise ctdb */
172         ctdb = ctdb_init(ev);
173         if (ctdb == NULL) {
174                 printf("Failed to init ctdb\n");
175                 exit(1);
176         }
177
178         if (self_connect) {
179                 ctdb_set_flags(ctdb, CTDB_FLAG_SELF_CONNECT);
180         }
181
182         ret = ctdb_set_transport(ctdb, transport);
183         if (ret == -1) {
184                 printf("ctdb_set_transport failed - %s\n", ctdb_errstr(ctdb));
185                 exit(1);
186         }
187
188         /* tell ctdb what address to listen on */
189         ret = ctdb_set_address(ctdb, myaddress);
190         if (ret == -1) {
191                 printf("ctdb_set_address failed - %s\n", ctdb_errstr(ctdb));
192                 exit(1);
193         }
194
195         /* tell ctdb what nodes are available */
196         ret = ctdb_set_nlist(ctdb, nlist);
197         if (ret == -1) {
198                 printf("ctdb_set_nlist failed - %s\n", ctdb_errstr(ctdb));
199                 exit(1);
200         }
201
202         /* setup a ctdb call function */
203         ret = ctdb_set_call(ctdb, incr_func,  FUNC_INCR);
204         ret = ctdb_set_call(ctdb, fetch_func, FUNC_FETCH);
205
206         /* attach to a specific database */
207         ret = ctdb_attach(ctdb, "test.tdb", TDB_DEFAULT, O_RDWR|O_CREAT|O_TRUNC, 0666);
208         if (ret == -1) {
209                 printf("ctdb_attach failed - %s\n", ctdb_errstr(ctdb));
210                 exit(1);
211         }
212
213         /* start the protocol running */
214         ret = ctdb_start(ctdb);
215
216         /* wait until all nodes are connected (should not be needed
217            outside of test code) */
218         ctdb_connect_wait(ctdb);
219
220         bench_incr(ctdb);
221        
222         /* go into a wait loop to allow other nodes to complete */
223         ctdb_wait_loop(ctdb);
224
225         /* shut it down */
226         talloc_free(ctdb);
227         return 0;
228 }