186d7f5d3SJohn Marino /*-
286d7f5d3SJohn Marino * Copyright (c) 2008 Peter Holm <pho@FreeBSD.org>
386d7f5d3SJohn Marino * All rights reserved.
486d7f5d3SJohn Marino *
586d7f5d3SJohn Marino * Redistribution and use in source and binary forms, with or without
686d7f5d3SJohn Marino * modification, are permitted provided that the following conditions
786d7f5d3SJohn Marino * are met:
886d7f5d3SJohn Marino * 1. Redistributions of source code must retain the above copyright
986d7f5d3SJohn Marino * notice, this list of conditions and the following disclaimer.
1086d7f5d3SJohn Marino * 2. Redistributions in binary form must reproduce the above copyright
1186d7f5d3SJohn Marino * notice, this list of conditions and the following disclaimer in the
1286d7f5d3SJohn Marino * documentation and/or other materials provided with the distribution.
1386d7f5d3SJohn Marino *
1486d7f5d3SJohn Marino * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
1586d7f5d3SJohn Marino * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
1686d7f5d3SJohn Marino * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
1786d7f5d3SJohn Marino * ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
1886d7f5d3SJohn Marino * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
1986d7f5d3SJohn Marino * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
2086d7f5d3SJohn Marino * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
2186d7f5d3SJohn Marino * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
2286d7f5d3SJohn Marino * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
2386d7f5d3SJohn Marino * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
2486d7f5d3SJohn Marino * SUCH DAMAGE.
2586d7f5d3SJohn Marino *
2686d7f5d3SJohn Marino */
2786d7f5d3SJohn Marino
2886d7f5d3SJohn Marino #include <stdio.h>
2986d7f5d3SJohn Marino #include <stdlib.h>
3086d7f5d3SJohn Marino #include <unistd.h>
3186d7f5d3SJohn Marino #include <sysexits.h>
3286d7f5d3SJohn Marino #include <string.h>
3386d7f5d3SJohn Marino #include <time.h>
3486d7f5d3SJohn Marino #include <err.h>
3586d7f5d3SJohn Marino
3686d7f5d3SJohn Marino #include "stress.h"
3786d7f5d3SJohn Marino
3886d7f5d3SJohn Marino opt_t opt;
3986d7f5d3SJohn Marino opt_t *op;
4086d7f5d3SJohn Marino
4186d7f5d3SJohn Marino static char path[64];
4286d7f5d3SJohn Marino
4386d7f5d3SJohn Marino static void
usage(char * where)4486d7f5d3SJohn Marino usage(char *where)
4586d7f5d3SJohn Marino {
4686d7f5d3SJohn Marino char *help;
4786d7f5d3SJohn Marino
4886d7f5d3SJohn Marino if (where != NULL)
4986d7f5d3SJohn Marino printf("Error in \"%s\"\n", where);
5086d7f5d3SJohn Marino fprintf(stderr, "Usage: %s [-t | -l | -i | -d | -h | -k | -v]\n", getprogname());
5186d7f5d3SJohn Marino help = " t <number><s|m|h|d> : time to run test\n"
5286d7f5d3SJohn Marino " l <pct> : load factor 0 - 100%\n"
5386d7f5d3SJohn Marino " i <number> : max # of parallel incarnations\n"
5486d7f5d3SJohn Marino " d <path> : working directory\n"
5586d7f5d3SJohn Marino " h : hog resources\n"
5686d7f5d3SJohn Marino " k : terminate with SIGHUP + SIGKILL\n"
5786d7f5d3SJohn Marino " n : no startup delay\n"
5886d7f5d3SJohn Marino " v : verbose\n";
5986d7f5d3SJohn Marino printf(help);
6086d7f5d3SJohn Marino exit(EX_USAGE);
6186d7f5d3SJohn Marino }
6286d7f5d3SJohn Marino
6386d7f5d3SJohn Marino static int
time2sec(const char * string)6486d7f5d3SJohn Marino time2sec(const char *string)
6586d7f5d3SJohn Marino {
6686d7f5d3SJohn Marino int r, s = 0;
6786d7f5d3SJohn Marino char modifier;
6886d7f5d3SJohn Marino r = sscanf(string, "%d%c", &s, &modifier);
6986d7f5d3SJohn Marino if (r == 2)
7086d7f5d3SJohn Marino switch(modifier) {
7186d7f5d3SJohn Marino case 's': break;
7286d7f5d3SJohn Marino case 'm': s = s * 60; break;
7386d7f5d3SJohn Marino case 'h': s = s * 60 * 60; break;
7486d7f5d3SJohn Marino case 'd': s = s * 60 * 60 * 24; break;
7586d7f5d3SJohn Marino default:
7686d7f5d3SJohn Marino usage("-t");
7786d7f5d3SJohn Marino }
7886d7f5d3SJohn Marino else
7986d7f5d3SJohn Marino usage("-t");
8086d7f5d3SJohn Marino return (s);
8186d7f5d3SJohn Marino }
8286d7f5d3SJohn Marino
gete(char * name)8386d7f5d3SJohn Marino static char *gete(char *name)
8486d7f5d3SJohn Marino {
8586d7f5d3SJohn Marino char *cp;
8686d7f5d3SJohn Marino char help[128];
8786d7f5d3SJohn Marino
8886d7f5d3SJohn Marino snprintf(help, sizeof(help), "%s%s", getprogname(), name);
8986d7f5d3SJohn Marino cp = getenv(help);
9086d7f5d3SJohn Marino if (cp == NULL)
9186d7f5d3SJohn Marino cp = getenv(name);
9286d7f5d3SJohn Marino return (cp);
9386d7f5d3SJohn Marino }
9486d7f5d3SJohn Marino
9586d7f5d3SJohn Marino static void
environment(void)9686d7f5d3SJohn Marino environment(void)
9786d7f5d3SJohn Marino {
9886d7f5d3SJohn Marino char *cp;
9986d7f5d3SJohn Marino
10086d7f5d3SJohn Marino if ((cp = gete("INCARNATIONS")) != NULL) {
10186d7f5d3SJohn Marino if (sscanf(cp, "%d", &op->incarnations) != 1)
10286d7f5d3SJohn Marino usage("INCARNATIONS");
10386d7f5d3SJohn Marino }
10486d7f5d3SJohn Marino if ((cp = gete("LOAD")) != NULL) {
10586d7f5d3SJohn Marino if (sscanf(cp, "%d", &op->load) != 1)
10686d7f5d3SJohn Marino usage("LOAD");
10786d7f5d3SJohn Marino }
10886d7f5d3SJohn Marino if ((cp = gete("RUNTIME")) != NULL) {
10986d7f5d3SJohn Marino op->run_time = time2sec(cp);
11086d7f5d3SJohn Marino }
11186d7f5d3SJohn Marino if ((cp = gete("RUNDIR")) != NULL) {
11286d7f5d3SJohn Marino op->wd = cp;
11386d7f5d3SJohn Marino }
11486d7f5d3SJohn Marino if ((cp = gete("CTRLDIR")) != NULL) {
11586d7f5d3SJohn Marino op->cd = cp;
11686d7f5d3SJohn Marino }
11786d7f5d3SJohn Marino if ((cp = gete("HOG")) != NULL) {
11886d7f5d3SJohn Marino op->hog = 1;
11986d7f5d3SJohn Marino }
12086d7f5d3SJohn Marino if ((cp = gete("KILL")) != NULL) {
12186d7f5d3SJohn Marino op->kill = 1;
12286d7f5d3SJohn Marino }
12386d7f5d3SJohn Marino if ((cp = gete("NODELAY")) != NULL) {
12486d7f5d3SJohn Marino op->nodelay = 1;
12586d7f5d3SJohn Marino }
12686d7f5d3SJohn Marino if ((cp = gete("VERBOSE")) != NULL) {
12786d7f5d3SJohn Marino if (sscanf(cp, "%d", &op->verbose) != 1)
12886d7f5d3SJohn Marino usage("VERBOSE");
12986d7f5d3SJohn Marino }
13086d7f5d3SJohn Marino if ((cp = gete("KBLOCKS")) != NULL) {
13186d7f5d3SJohn Marino if (sscanf(cp, "%d", &op->kblocks) != 1)
13286d7f5d3SJohn Marino usage("KBLOCKS");
13386d7f5d3SJohn Marino }
13486d7f5d3SJohn Marino if ((cp = gete("INODES")) != NULL) {
13586d7f5d3SJohn Marino if (sscanf(cp, "%d", &op->inodes) != 1)
13686d7f5d3SJohn Marino usage("INODES");
13786d7f5d3SJohn Marino }
13886d7f5d3SJohn Marino }
13986d7f5d3SJohn Marino
14086d7f5d3SJohn Marino void
options(int argc,char ** argv)14186d7f5d3SJohn Marino options(int argc, char **argv)
14286d7f5d3SJohn Marino {
14386d7f5d3SJohn Marino int ch;
14486d7f5d3SJohn Marino
14586d7f5d3SJohn Marino op = &opt;
14686d7f5d3SJohn Marino
14786d7f5d3SJohn Marino op->run_time = 60;
14886d7f5d3SJohn Marino op->load = 100;
14986d7f5d3SJohn Marino op->wd = "/tmp/stressX";
15086d7f5d3SJohn Marino op->cd = "/tmp/stressX.control";
15186d7f5d3SJohn Marino op->incarnations = 1;
15286d7f5d3SJohn Marino op->hog = 0;
15386d7f5d3SJohn Marino op->kill = 0;
15486d7f5d3SJohn Marino op->nodelay = 0;
15586d7f5d3SJohn Marino op->verbose = 0;
15686d7f5d3SJohn Marino op->kblocks = 0;
15786d7f5d3SJohn Marino op->inodes = 0;
15886d7f5d3SJohn Marino
15986d7f5d3SJohn Marino environment();
16086d7f5d3SJohn Marino
16186d7f5d3SJohn Marino while ((ch = getopt(argc, argv, "t:l:i:d:hknv")) != -1)
16286d7f5d3SJohn Marino switch(ch) {
16386d7f5d3SJohn Marino case 't': /* run time */
16486d7f5d3SJohn Marino op->run_time = time2sec(optarg);
16586d7f5d3SJohn Marino break;
16686d7f5d3SJohn Marino case 'l': /* load factor in pct */
16786d7f5d3SJohn Marino if (sscanf(optarg, "%d", &op->load) != 1)
16886d7f5d3SJohn Marino usage("-l");
16986d7f5d3SJohn Marino break;
17086d7f5d3SJohn Marino case 'i': /* max incarnations */
17186d7f5d3SJohn Marino if (sscanf(optarg, "%d", &op->incarnations) != 1)
17286d7f5d3SJohn Marino usage("-i");
17386d7f5d3SJohn Marino break;
17486d7f5d3SJohn Marino case 'd': /* working directory */
17586d7f5d3SJohn Marino op->wd = strdup(optarg);
17686d7f5d3SJohn Marino break;
17786d7f5d3SJohn Marino case 'h': /* hog flag */
17886d7f5d3SJohn Marino op->hog += 1;
17986d7f5d3SJohn Marino break;
18086d7f5d3SJohn Marino case 'k': /* kill flag */
18186d7f5d3SJohn Marino op->kill = 1;
18286d7f5d3SJohn Marino break;
18386d7f5d3SJohn Marino case 'n': /* no delay flag */
18486d7f5d3SJohn Marino op->nodelay = 1;
18586d7f5d3SJohn Marino break;
18686d7f5d3SJohn Marino case 'v': /* verbose flag */
18786d7f5d3SJohn Marino op->verbose += 1;
18886d7f5d3SJohn Marino break;
18986d7f5d3SJohn Marino default:
19086d7f5d3SJohn Marino usage(NULL);
19186d7f5d3SJohn Marino }
19286d7f5d3SJohn Marino op->argc = argc -= optind;
19386d7f5d3SJohn Marino op->argv = argv += optind;
19486d7f5d3SJohn Marino
19586d7f5d3SJohn Marino if (op->incarnations < 1)
19686d7f5d3SJohn Marino op->incarnations = 1;
19786d7f5d3SJohn Marino if (op->hog == 0)
19886d7f5d3SJohn Marino op->incarnations = random_int(1, op->incarnations);
19986d7f5d3SJohn Marino if (op->run_time < 15)
20086d7f5d3SJohn Marino op->run_time = 15;
20186d7f5d3SJohn Marino if (op->load < 0 || op->load > 100)
20286d7f5d3SJohn Marino op->load = 100;
20386d7f5d3SJohn Marino }
20486d7f5d3SJohn Marino
20586d7f5d3SJohn Marino void
show_status(void)20686d7f5d3SJohn Marino show_status(void)
20786d7f5d3SJohn Marino {
20886d7f5d3SJohn Marino char buf[80];
20986d7f5d3SJohn Marino int days;
21086d7f5d3SJohn Marino time_t t;
21186d7f5d3SJohn Marino
21286d7f5d3SJohn Marino if (op->verbose > 0) {
21386d7f5d3SJohn Marino t = op->run_time;
21486d7f5d3SJohn Marino days = t / (60 * 60 * 24);
21586d7f5d3SJohn Marino t = t % (60 * 60 * 24);
21686d7f5d3SJohn Marino strftime(buf, sizeof(buf), "%T", gmtime(&t));
21786d7f5d3SJohn Marino printf("%8s: run time %2d+%s, incarnations %3d, load %3d, verbose %d\n",
21886d7f5d3SJohn Marino getprogname(), days, buf, op->incarnations, op->load,
21986d7f5d3SJohn Marino op->verbose);
22086d7f5d3SJohn Marino }
22186d7f5d3SJohn Marino }
22286d7f5d3SJohn Marino
22386d7f5d3SJohn Marino void
rmval(void)22486d7f5d3SJohn Marino rmval(void)
22586d7f5d3SJohn Marino {
22686d7f5d3SJohn Marino if (snprintf(path, sizeof(path), "%s/%s.conf", op->cd, getprogname()) < 0)
22786d7f5d3SJohn Marino err(1, "snprintf path");
22886d7f5d3SJohn Marino (void) unlink(path);
22986d7f5d3SJohn Marino }
23086d7f5d3SJohn Marino
23186d7f5d3SJohn Marino void
putval(unsigned long v)23286d7f5d3SJohn Marino putval(unsigned long v)
23386d7f5d3SJohn Marino {
23486d7f5d3SJohn Marino char buf[64];
23586d7f5d3SJohn Marino
23686d7f5d3SJohn Marino rmval();
23786d7f5d3SJohn Marino snprintf(buf, sizeof(buf), "%lu", v);
23886d7f5d3SJohn Marino if (symlink(buf, path) < 0)
23986d7f5d3SJohn Marino err(1, "symlink(%s, %s)", path, buf);
24086d7f5d3SJohn Marino }
24186d7f5d3SJohn Marino
24286d7f5d3SJohn Marino unsigned long
getval(void)24386d7f5d3SJohn Marino getval(void)
24486d7f5d3SJohn Marino {
24586d7f5d3SJohn Marino int i, n;
24686d7f5d3SJohn Marino unsigned long val;
24786d7f5d3SJohn Marino char buf[64];
24886d7f5d3SJohn Marino
24986d7f5d3SJohn Marino if ((n = readlink(path, buf, sizeof(buf) -1)) < 0) {
25086d7f5d3SJohn Marino for (i = 0; i < 60; i++) {
25186d7f5d3SJohn Marino sleep(1);
25286d7f5d3SJohn Marino if ((n = readlink(path, buf, sizeof(buf) -1)) > 0)
25386d7f5d3SJohn Marino break;
25486d7f5d3SJohn Marino }
25586d7f5d3SJohn Marino if (n < 0)
25686d7f5d3SJohn Marino err(1, "readlink(%s). %s:%d", path, __FILE__, __LINE__);
25786d7f5d3SJohn Marino }
25886d7f5d3SJohn Marino buf[n] = '\0';
25986d7f5d3SJohn Marino if (sscanf(buf, "%ld", &val) != 1)
26086d7f5d3SJohn Marino err(1, "sscanf(%s)", buf);
26186d7f5d3SJohn Marino return val;
26286d7f5d3SJohn Marino }
263