/*
dbench version 3
Copyright (C) Andrew Tridgell 1999-2004
This program is free software; you can redistribute it and/or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation; either version 3 of the License, or
(at your option) any later version.
This program is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU General Public License for more details.
You should have received a copy of the GNU General Public License
along with this program; if not, see .
*/
/* This file links against either fileio.c to do operations against a
local filesystem (making dbench), or sockio.c to issue SMB-like
command packets over a socket (making tbench).
So, the pattern of operations and the control structure is the same
for both benchmarks, but the operations performed are different.
*/
#include "dbench.h"
#define ival(s) strtol(s, NULL, 0)
static void nb_target_rate(struct child_struct *child, double rate)
{
double tdelay;
if (child->rate.last_bytes == 0) {
child->rate.last_bytes = child->bytes;
child->rate.last_time = timeval_current();
return;
}
if (rate != 0) {
tdelay = (child->bytes - child->rate.last_bytes)/(1.0e6*rate) -
timeval_elapsed(&child->rate.last_time);
} else {
tdelay = - timeval_elapsed(&child->rate.last_time);
}
if (tdelay > 0 && rate != 0) {
msleep(tdelay*1000);
} else {
child->max_latency = MAX(child->max_latency, -tdelay);
}
child->rate.last_time = timeval_current();
child->rate.last_bytes = child->bytes;
}
static void nb_time_reset(struct child_struct *child)
{
child->starttime = timeval_current();
memset(&child->rate, 0, sizeof(child->rate));
}
static void nb_time_delay(struct child_struct *child, double targett)
{
double elapsed = timeval_elapsed(&child->starttime);
if (targett > elapsed) {
msleep(1000*(targett - elapsed));
} else if (elapsed - targett > child->max_latency) {
child->max_latency = MAX(elapsed - targett, child->max_latency);
}
}
static void finish_op(struct child_struct *child, struct op *op)
{
double t = timeval_elapsed(&child->lasttime);
op->count++;
op->total_time += t;
if (t > op->max_latency) {
op->max_latency = t;
}
}
#define OP_LATENCY(opname) finish_op(child, &child->op.op_ ## opname)
/*
one child operation
*/
static void child_op(struct child_struct *child, char **params,
const char *fname, const char *fname2, const char *status)
{
child->lasttime = timeval_current();
if (!strcmp(params[0],"NTCreateX")) {
nb_createx(child, fname, ival(params[2]), ival(params[3]),
ival(params[4]), status);
OP_LATENCY(NTCreateX);
} else if (!strcmp(params[0],"Close")) {
nb_close(child, ival(params[1]), status);
OP_LATENCY(Close);
} else if (!strcmp(params[0],"Rename")) {
nb_rename(child, fname, fname2, status);
OP_LATENCY(Rename);
} else if (!strcmp(params[0],"Unlink")) {
nb_unlink(child, fname, ival(params[2]), status);
OP_LATENCY(Unlink);
} else if (!strcmp(params[0],"Deltree")) {
nb_deltree(child, fname);
OP_LATENCY(Deltree);
} else if (!strcmp(params[0],"Rmdir")) {
nb_rmdir(child, fname, status);
OP_LATENCY(Rmdir);
} else if (!strcmp(params[0],"Mkdir")) {
nb_mkdir(child, fname, status);
OP_LATENCY(Mkdir);
} else if (!strcmp(params[0],"QUERY_PATH_INFORMATION")) {
nb_qpathinfo(child, fname, ival(params[2]), status);
OP_LATENCY(Qpathinfo);
} else if (!strcmp(params[0],"QUERY_FILE_INFORMATION")) {
nb_qfileinfo(child, ival(params[1]), ival(params[2]), status);
OP_LATENCY(Qfileinfo);
} else if (!strcmp(params[0],"QUERY_FS_INFORMATION")) {
nb_qfsinfo(child, ival(params[1]), status);
OP_LATENCY(Qfsinfo);
} else if (!strcmp(params[0],"SET_FILE_INFORMATION")) {
nb_sfileinfo(child, ival(params[1]), ival(params[2]), status);
OP_LATENCY(Sfileinfo);
} else if (!strcmp(params[0],"FIND_FIRST")) {
nb_findfirst(child, fname, ival(params[2]),
ival(params[3]), ival(params[4]), status);
OP_LATENCY(Find);
} else if (!strcmp(params[0],"WriteX")) {
nb_writex(child, ival(params[1]),
ival(params[2]), ival(params[3]), ival(params[4]),
status);
OP_LATENCY(WriteX);
} else if (!strcmp(params[0],"LockX")) {
nb_lockx(child, ival(params[1]),
ival(params[2]), ival(params[3]), status);
OP_LATENCY(LockX);
} else if (!strcmp(params[0],"UnlockX")) {
nb_unlockx(child, ival(params[1]),
ival(params[2]), ival(params[3]), status);
OP_LATENCY(UnlockX);
} else if (!strcmp(params[0],"ReadX")) {
nb_readx(child, ival(params[1]),
ival(params[2]), ival(params[3]), ival(params[4]),
status);
OP_LATENCY(ReadX);
} else if (!strcmp(params[0],"Flush")) {
nb_flush(child, ival(params[1]), status);
OP_LATENCY(Flush);
} else if (!strcmp(params[0],"Sleep")) {
nb_sleep(child, ival(params[1]), status);
} else {
printf("[%d] Unknown operation %s in pid %d\n",
child->line, params[0], getpid());
}
}
/* run a test that simulates an approximate netbench client load */
void child_run(struct child_struct *child0, const char *loadfile)
{
int i;
char line[1024], fname[1024], fname2[1024];
char **sparams, **params;
char *p;
const char *status;
FILE *f;
pid_t parent = getppid();
double targett;
struct child_struct *child;
for (child=child0;childline = 0;
asprintf(&child->cname,"client%d", child->id);
}
sparams = calloc(20, sizeof(char *));
for (i=0;i<20;i++) {
sparams[i] = malloc(100);
}
f = fopen(loadfile, "r");
if (f == NULL) {
perror(loadfile);
exit(1);
}
again:
for (child=child0;childdone) goto done;
child->line++;
}
line[strlen(line)-1] = 0;
all_string_sub(line,"\\", "/");
all_string_sub(line," /", " ");
p = line;
for (i=0;
i<19 && next_token(&p, params[i], " ");
i++) ;
params[i][0] = 0;
if (i < 2 || params[0][0] == '#') continue;
if (!strncmp(params[0],"SMB", 3)) {
printf("ERROR: You are using a dbench 1 load file\n");
exit(1);
}
if (i > 0 && isdigit(params[0][0])) {
targett = strtod(params[0], NULL);
params++;
i--;
} else {
targett = 0.0;
}
if (strncmp(params[i-1], "NT_STATUS_", 10) != 0 &&
strncmp(params[i-1], "0x", 2) != 0) {
printf("Badly formed status at line %d\n", child->line);
continue;
}
status = params[i-1];
for (child=child0;child1 && params[1][0] == '/') {
snprintf(fname, sizeof(fname), "%s%s", child->directory, params[1]);
all_string_sub(fname,"client1", child->cname);
}
if (i>2 && params[2][0] == '/') {
snprintf(fname2, sizeof(fname2), "%s%s", child->directory, params[2]);
all_string_sub(fname2,"client1", child->cname);
}
if (options.targetrate != 0 || targett == 0.0) {
nb_target_rate(child, options.targetrate);
} else {
nb_time_delay(child, targett);
}
child_op(child, params, fname, fname2, status);
}
}
rewind(f);
goto again;
done:
fclose(f);
for (child=child0;childcleanup = 1;
fflush(stdout);
if (!options.skip_cleanup) {
nb_cleanup(child);
}
child->cleanup_finished = 1;
}
}