1*212c1933SFabiano Rosas /*
2*212c1933SFabiano Rosas * Migration stress workload
3*212c1933SFabiano Rosas *
4*212c1933SFabiano Rosas * Copyright (c) 2016 Red Hat, Inc.
5*212c1933SFabiano Rosas *
6*212c1933SFabiano Rosas * This library is free software; you can redistribute it and/or
7*212c1933SFabiano Rosas * modify it under the terms of the GNU Lesser General Public
8*212c1933SFabiano Rosas * License as published by the Free Software Foundation; either
9*212c1933SFabiano Rosas * version 2.1 of the License, or (at your option) any later version.
10*212c1933SFabiano Rosas *
11*212c1933SFabiano Rosas * This library is distributed in the hope that it will be useful,
12*212c1933SFabiano Rosas * but WITHOUT ANY WARRANTY; without even the implied warranty of
13*212c1933SFabiano Rosas * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
14*212c1933SFabiano Rosas * Lesser General Public License for more details.
15*212c1933SFabiano Rosas *
16*212c1933SFabiano Rosas * You should have received a copy of the GNU Lesser General Public
17*212c1933SFabiano Rosas * License along with this library; if not, see <http://www.gnu.org/licenses/>.
18*212c1933SFabiano Rosas */
19*212c1933SFabiano Rosas
20*212c1933SFabiano Rosas #include "qemu/osdep.h"
21*212c1933SFabiano Rosas #include <getopt.h>
22*212c1933SFabiano Rosas #include <sys/reboot.h>
23*212c1933SFabiano Rosas #include <sys/syscall.h>
24*212c1933SFabiano Rosas #include <linux/random.h>
25*212c1933SFabiano Rosas #include <pthread.h>
26*212c1933SFabiano Rosas #include <sys/mount.h>
27*212c1933SFabiano Rosas
28*212c1933SFabiano Rosas const char *argv0;
29*212c1933SFabiano Rosas
30*212c1933SFabiano Rosas #define RAM_PAGE_SIZE 4096
31*212c1933SFabiano Rosas
32*212c1933SFabiano Rosas #ifndef CONFIG_GETTID
gettid(void)33*212c1933SFabiano Rosas static int gettid(void)
34*212c1933SFabiano Rosas {
35*212c1933SFabiano Rosas return syscall(SYS_gettid);
36*212c1933SFabiano Rosas }
37*212c1933SFabiano Rosas #endif
38*212c1933SFabiano Rosas
exit_failure(void)39*212c1933SFabiano Rosas static __attribute__((noreturn)) void exit_failure(void)
40*212c1933SFabiano Rosas {
41*212c1933SFabiano Rosas if (getpid() == 1) {
42*212c1933SFabiano Rosas sync();
43*212c1933SFabiano Rosas reboot(RB_POWER_OFF);
44*212c1933SFabiano Rosas fprintf(stderr, "%s (%05d): ERROR: cannot reboot: %s\n",
45*212c1933SFabiano Rosas argv0, gettid(), strerror(errno));
46*212c1933SFabiano Rosas abort();
47*212c1933SFabiano Rosas } else {
48*212c1933SFabiano Rosas exit(1);
49*212c1933SFabiano Rosas }
50*212c1933SFabiano Rosas }
51*212c1933SFabiano Rosas
get_command_arg_str(const char * name,char ** val)52*212c1933SFabiano Rosas static int get_command_arg_str(const char *name,
53*212c1933SFabiano Rosas char **val)
54*212c1933SFabiano Rosas {
55*212c1933SFabiano Rosas static char line[1024];
56*212c1933SFabiano Rosas FILE *fp = fopen("/proc/cmdline", "r");
57*212c1933SFabiano Rosas char *start, *end;
58*212c1933SFabiano Rosas
59*212c1933SFabiano Rosas if (fp == NULL) {
60*212c1933SFabiano Rosas fprintf(stderr, "%s (%05d): ERROR: cannot open /proc/cmdline: %s\n",
61*212c1933SFabiano Rosas argv0, gettid(), strerror(errno));
62*212c1933SFabiano Rosas return -1;
63*212c1933SFabiano Rosas }
64*212c1933SFabiano Rosas
65*212c1933SFabiano Rosas if (!fgets(line, sizeof line, fp)) {
66*212c1933SFabiano Rosas fprintf(stderr, "%s (%05d): ERROR: cannot read /proc/cmdline: %s\n",
67*212c1933SFabiano Rosas argv0, gettid(), strerror(errno));
68*212c1933SFabiano Rosas fclose(fp);
69*212c1933SFabiano Rosas return -1;
70*212c1933SFabiano Rosas }
71*212c1933SFabiano Rosas fclose(fp);
72*212c1933SFabiano Rosas
73*212c1933SFabiano Rosas start = strstr(line, name);
74*212c1933SFabiano Rosas if (!start)
75*212c1933SFabiano Rosas return 0;
76*212c1933SFabiano Rosas
77*212c1933SFabiano Rosas start += strlen(name);
78*212c1933SFabiano Rosas
79*212c1933SFabiano Rosas if (*start != '=') {
80*212c1933SFabiano Rosas fprintf(stderr, "%s (%05d): ERROR: no value provided for '%s' in /proc/cmdline\n",
81*212c1933SFabiano Rosas argv0, gettid(), name);
82*212c1933SFabiano Rosas }
83*212c1933SFabiano Rosas start++;
84*212c1933SFabiano Rosas
85*212c1933SFabiano Rosas end = strstr(start, " ");
86*212c1933SFabiano Rosas if (!end)
87*212c1933SFabiano Rosas end = strstr(start, "\n");
88*212c1933SFabiano Rosas
89*212c1933SFabiano Rosas if (end == start) {
90*212c1933SFabiano Rosas fprintf(stderr, "%s (%05d): ERROR: no value provided for '%s' in /proc/cmdline\n",
91*212c1933SFabiano Rosas argv0, gettid(), name);
92*212c1933SFabiano Rosas return -1;
93*212c1933SFabiano Rosas }
94*212c1933SFabiano Rosas
95*212c1933SFabiano Rosas if (end)
96*212c1933SFabiano Rosas *val = g_strndup(start, end - start);
97*212c1933SFabiano Rosas else
98*212c1933SFabiano Rosas *val = g_strdup(start);
99*212c1933SFabiano Rosas return 1;
100*212c1933SFabiano Rosas }
101*212c1933SFabiano Rosas
102*212c1933SFabiano Rosas
get_command_arg_ull(const char * name,unsigned long long * val)103*212c1933SFabiano Rosas static int get_command_arg_ull(const char *name,
104*212c1933SFabiano Rosas unsigned long long *val)
105*212c1933SFabiano Rosas {
106*212c1933SFabiano Rosas char *valstr;
107*212c1933SFabiano Rosas char *end;
108*212c1933SFabiano Rosas
109*212c1933SFabiano Rosas int ret = get_command_arg_str(name, &valstr);
110*212c1933SFabiano Rosas if (ret <= 0)
111*212c1933SFabiano Rosas return ret;
112*212c1933SFabiano Rosas
113*212c1933SFabiano Rosas errno = 0;
114*212c1933SFabiano Rosas *val = strtoll(valstr, &end, 10);
115*212c1933SFabiano Rosas if (errno || *end) {
116*212c1933SFabiano Rosas fprintf(stderr, "%s (%05d): ERROR: cannot parse %s value %s\n",
117*212c1933SFabiano Rosas argv0, gettid(), name, valstr);
118*212c1933SFabiano Rosas g_free(valstr);
119*212c1933SFabiano Rosas return -1;
120*212c1933SFabiano Rosas }
121*212c1933SFabiano Rosas g_free(valstr);
122*212c1933SFabiano Rosas return 0;
123*212c1933SFabiano Rosas }
124*212c1933SFabiano Rosas
125*212c1933SFabiano Rosas
random_bytes(char * buf,size_t len)126*212c1933SFabiano Rosas static int random_bytes(char *buf, size_t len)
127*212c1933SFabiano Rosas {
128*212c1933SFabiano Rosas int fd;
129*212c1933SFabiano Rosas
130*212c1933SFabiano Rosas fd = open("/dev/urandom", O_RDONLY);
131*212c1933SFabiano Rosas if (fd < 0) {
132*212c1933SFabiano Rosas fprintf(stderr, "%s (%05d): ERROR: cannot open /dev/urandom: %s\n",
133*212c1933SFabiano Rosas argv0, gettid(), strerror(errno));
134*212c1933SFabiano Rosas return -1;
135*212c1933SFabiano Rosas }
136*212c1933SFabiano Rosas
137*212c1933SFabiano Rosas if (read(fd, buf, len) != len) {
138*212c1933SFabiano Rosas fprintf(stderr, "%s (%05d): ERROR: cannot read /dev/urandom: %s\n",
139*212c1933SFabiano Rosas argv0, gettid(), strerror(errno));
140*212c1933SFabiano Rosas close(fd);
141*212c1933SFabiano Rosas return -1;
142*212c1933SFabiano Rosas }
143*212c1933SFabiano Rosas
144*212c1933SFabiano Rosas close(fd);
145*212c1933SFabiano Rosas
146*212c1933SFabiano Rosas return 0;
147*212c1933SFabiano Rosas }
148*212c1933SFabiano Rosas
149*212c1933SFabiano Rosas
now(void)150*212c1933SFabiano Rosas static unsigned long long now(void)
151*212c1933SFabiano Rosas {
152*212c1933SFabiano Rosas struct timeval tv;
153*212c1933SFabiano Rosas
154*212c1933SFabiano Rosas gettimeofday(&tv, NULL);
155*212c1933SFabiano Rosas
156*212c1933SFabiano Rosas return (tv.tv_sec * 1000ull) + (tv.tv_usec / 1000ull);
157*212c1933SFabiano Rosas }
158*212c1933SFabiano Rosas
stressone(unsigned long long ramsizeMB)159*212c1933SFabiano Rosas static void stressone(unsigned long long ramsizeMB)
160*212c1933SFabiano Rosas {
161*212c1933SFabiano Rosas size_t pagesPerMB = 1024 * 1024 / RAM_PAGE_SIZE;
162*212c1933SFabiano Rosas g_autofree char *ram = g_malloc(ramsizeMB * 1024 * 1024);
163*212c1933SFabiano Rosas char *ramptr;
164*212c1933SFabiano Rosas size_t i, j, k;
165*212c1933SFabiano Rosas g_autofree char *data = g_malloc(RAM_PAGE_SIZE);
166*212c1933SFabiano Rosas char *dataptr;
167*212c1933SFabiano Rosas size_t nMB = 0;
168*212c1933SFabiano Rosas unsigned long long before, after;
169*212c1933SFabiano Rosas
170*212c1933SFabiano Rosas /* We don't care about initial state, but we do want
171*212c1933SFabiano Rosas * to fault it all into RAM, otherwise the first iter
172*212c1933SFabiano Rosas * of the loop below will be quite slow. We can't use
173*212c1933SFabiano Rosas * 0x0 as the byte as gcc optimizes that away into a
174*212c1933SFabiano Rosas * calloc instead :-) */
175*212c1933SFabiano Rosas memset(ram, 0xfe, ramsizeMB * 1024 * 1024);
176*212c1933SFabiano Rosas
177*212c1933SFabiano Rosas if (random_bytes(data, RAM_PAGE_SIZE) < 0) {
178*212c1933SFabiano Rosas return;
179*212c1933SFabiano Rosas }
180*212c1933SFabiano Rosas
181*212c1933SFabiano Rosas before = now();
182*212c1933SFabiano Rosas
183*212c1933SFabiano Rosas while (1) {
184*212c1933SFabiano Rosas
185*212c1933SFabiano Rosas ramptr = ram;
186*212c1933SFabiano Rosas for (i = 0; i < ramsizeMB; i++, nMB++) {
187*212c1933SFabiano Rosas for (j = 0; j < pagesPerMB; j++) {
188*212c1933SFabiano Rosas dataptr = data;
189*212c1933SFabiano Rosas for (k = 0; k < RAM_PAGE_SIZE; k += sizeof(long long)) {
190*212c1933SFabiano Rosas ramptr += sizeof(long long);
191*212c1933SFabiano Rosas dataptr += sizeof(long long);
192*212c1933SFabiano Rosas *(unsigned long long *)ramptr ^= *(unsigned long long *)dataptr;
193*212c1933SFabiano Rosas }
194*212c1933SFabiano Rosas }
195*212c1933SFabiano Rosas
196*212c1933SFabiano Rosas if (nMB == 1024) {
197*212c1933SFabiano Rosas after = now();
198*212c1933SFabiano Rosas fprintf(stderr, "%s (%05d): INFO: %06llums copied 1 GB in %05llums\n",
199*212c1933SFabiano Rosas argv0, gettid(), after, after - before);
200*212c1933SFabiano Rosas before = now();
201*212c1933SFabiano Rosas nMB = 0;
202*212c1933SFabiano Rosas }
203*212c1933SFabiano Rosas }
204*212c1933SFabiano Rosas }
205*212c1933SFabiano Rosas }
206*212c1933SFabiano Rosas
207*212c1933SFabiano Rosas
stressthread(void * arg)208*212c1933SFabiano Rosas static void *stressthread(void *arg)
209*212c1933SFabiano Rosas {
210*212c1933SFabiano Rosas unsigned long long ramsizeMB = *(unsigned long long *)arg;
211*212c1933SFabiano Rosas
212*212c1933SFabiano Rosas stressone(ramsizeMB);
213*212c1933SFabiano Rosas
214*212c1933SFabiano Rosas return NULL;
215*212c1933SFabiano Rosas }
216*212c1933SFabiano Rosas
stress(unsigned long long ramsizeGB,int ncpus)217*212c1933SFabiano Rosas static void stress(unsigned long long ramsizeGB, int ncpus)
218*212c1933SFabiano Rosas {
219*212c1933SFabiano Rosas size_t i;
220*212c1933SFabiano Rosas unsigned long long ramsizeMB = ramsizeGB * 1024 / ncpus;
221*212c1933SFabiano Rosas ncpus--;
222*212c1933SFabiano Rosas
223*212c1933SFabiano Rosas for (i = 0; i < ncpus; i++) {
224*212c1933SFabiano Rosas pthread_t thr;
225*212c1933SFabiano Rosas pthread_create(&thr, NULL,
226*212c1933SFabiano Rosas stressthread, &ramsizeMB);
227*212c1933SFabiano Rosas }
228*212c1933SFabiano Rosas
229*212c1933SFabiano Rosas stressone(ramsizeMB);
230*212c1933SFabiano Rosas }
231*212c1933SFabiano Rosas
232*212c1933SFabiano Rosas
mount_misc(const char * fstype,const char * dir)233*212c1933SFabiano Rosas static int mount_misc(const char *fstype, const char *dir)
234*212c1933SFabiano Rosas {
235*212c1933SFabiano Rosas if (g_mkdir_with_parents(dir, 0755) < 0 && errno != EEXIST) {
236*212c1933SFabiano Rosas fprintf(stderr, "%s (%05d): ERROR: cannot create %s: %s\n",
237*212c1933SFabiano Rosas argv0, gettid(), dir, strerror(errno));
238*212c1933SFabiano Rosas return -1;
239*212c1933SFabiano Rosas }
240*212c1933SFabiano Rosas
241*212c1933SFabiano Rosas if (mount("none", dir, fstype, 0, NULL) < 0) {
242*212c1933SFabiano Rosas fprintf(stderr, "%s (%05d): ERROR: cannot mount %s: %s\n",
243*212c1933SFabiano Rosas argv0, gettid(), dir, strerror(errno));
244*212c1933SFabiano Rosas return -1;
245*212c1933SFabiano Rosas }
246*212c1933SFabiano Rosas
247*212c1933SFabiano Rosas return 0;
248*212c1933SFabiano Rosas }
249*212c1933SFabiano Rosas
mount_all(void)250*212c1933SFabiano Rosas static int mount_all(void)
251*212c1933SFabiano Rosas {
252*212c1933SFabiano Rosas if (mount_misc("proc", "/proc") < 0 ||
253*212c1933SFabiano Rosas mount_misc("sysfs", "/sys") < 0 ||
254*212c1933SFabiano Rosas mount_misc("tmpfs", "/dev") < 0)
255*212c1933SFabiano Rosas return -1;
256*212c1933SFabiano Rosas
257*212c1933SFabiano Rosas mknod("/dev/urandom", 0777 | S_IFCHR, makedev(1, 9));
258*212c1933SFabiano Rosas mknod("/dev/random", 0777 | S_IFCHR, makedev(1, 8));
259*212c1933SFabiano Rosas
260*212c1933SFabiano Rosas return 0;
261*212c1933SFabiano Rosas }
262*212c1933SFabiano Rosas
main(int argc,char ** argv)263*212c1933SFabiano Rosas int main(int argc, char **argv)
264*212c1933SFabiano Rosas {
265*212c1933SFabiano Rosas unsigned long long ramsizeGB = 1;
266*212c1933SFabiano Rosas char *end;
267*212c1933SFabiano Rosas int ch;
268*212c1933SFabiano Rosas int opt_ind = 0;
269*212c1933SFabiano Rosas const char *sopt = "hr:c:";
270*212c1933SFabiano Rosas struct option lopt[] = {
271*212c1933SFabiano Rosas { "help", no_argument, NULL, 'h' },
272*212c1933SFabiano Rosas { "ramsize", required_argument, NULL, 'r' },
273*212c1933SFabiano Rosas { "cpus", required_argument, NULL, 'c' },
274*212c1933SFabiano Rosas { NULL, 0, NULL, 0 }
275*212c1933SFabiano Rosas };
276*212c1933SFabiano Rosas int ret;
277*212c1933SFabiano Rosas int ncpus = 0;
278*212c1933SFabiano Rosas
279*212c1933SFabiano Rosas argv0 = argv[0];
280*212c1933SFabiano Rosas
281*212c1933SFabiano Rosas while ((ch = getopt_long(argc, argv, sopt, lopt, &opt_ind)) != -1) {
282*212c1933SFabiano Rosas switch (ch) {
283*212c1933SFabiano Rosas case 'r':
284*212c1933SFabiano Rosas errno = 0;
285*212c1933SFabiano Rosas ramsizeGB = strtoll(optarg, &end, 10);
286*212c1933SFabiano Rosas if (errno != 0 || *end) {
287*212c1933SFabiano Rosas fprintf(stderr, "%s (%05d): ERROR: Cannot parse RAM size %s\n",
288*212c1933SFabiano Rosas argv0, gettid(), optarg);
289*212c1933SFabiano Rosas exit_failure();
290*212c1933SFabiano Rosas }
291*212c1933SFabiano Rosas break;
292*212c1933SFabiano Rosas
293*212c1933SFabiano Rosas case 'c':
294*212c1933SFabiano Rosas errno = 0;
295*212c1933SFabiano Rosas ncpus = strtoll(optarg, &end, 10);
296*212c1933SFabiano Rosas if (errno != 0 || *end) {
297*212c1933SFabiano Rosas fprintf(stderr, "%s (%05d): ERROR: Cannot parse CPU count %s\n",
298*212c1933SFabiano Rosas argv0, gettid(), optarg);
299*212c1933SFabiano Rosas exit_failure();
300*212c1933SFabiano Rosas }
301*212c1933SFabiano Rosas break;
302*212c1933SFabiano Rosas
303*212c1933SFabiano Rosas case '?':
304*212c1933SFabiano Rosas case 'h':
305*212c1933SFabiano Rosas fprintf(stderr, "%s: [--help][--ramsize GB][--cpus N]\n", argv0);
306*212c1933SFabiano Rosas exit_failure();
307*212c1933SFabiano Rosas }
308*212c1933SFabiano Rosas }
309*212c1933SFabiano Rosas
310*212c1933SFabiano Rosas if (getpid() == 1) {
311*212c1933SFabiano Rosas if (mount_all() < 0)
312*212c1933SFabiano Rosas exit_failure();
313*212c1933SFabiano Rosas
314*212c1933SFabiano Rosas ret = get_command_arg_ull("ramsize", &ramsizeGB);
315*212c1933SFabiano Rosas if (ret < 0)
316*212c1933SFabiano Rosas exit_failure();
317*212c1933SFabiano Rosas }
318*212c1933SFabiano Rosas
319*212c1933SFabiano Rosas if (ncpus == 0)
320*212c1933SFabiano Rosas ncpus = sysconf(_SC_NPROCESSORS_ONLN);
321*212c1933SFabiano Rosas
322*212c1933SFabiano Rosas fprintf(stdout, "%s (%05d): INFO: RAM %llu GiB across %d CPUs\n",
323*212c1933SFabiano Rosas argv0, gettid(), ramsizeGB, ncpus);
324*212c1933SFabiano Rosas
325*212c1933SFabiano Rosas stress(ramsizeGB, ncpus);
326*212c1933SFabiano Rosas
327*212c1933SFabiano Rosas exit_failure();
328*212c1933SFabiano Rosas }
329