1 /*****************************************************************************/
2 /* */
3 /* Copyright (c) 2010 Mohamed Naufal Basheer */
4 /* */
5 /* This program is free software; you can redistribute it and/or modify */
6 /* it under the terms of the GNU General Public License as published by */
7 /* the Free Software Foundation; either version 2 of the License, or */
8 /* (at your option) any later version. */
9 /* */
10 /* This program is distributed in the hope that it will be useful, */
11 /* but WITHOUT ANY WARRANTY; without even the implied warranty of */
12 /* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See */
13 /* the GNU General Public License for more details. */
14 /* */
15 /* You should have received a copy of the GNU General Public License */
16 /* along with this program; if not, write to the Free Software */
17 /* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA */
18 /* */
19 /* File: mem_process.c */
20 /* */
21 /* Purpose: act as a memory hog for the memcg_control tests */
22 /* */
23 /* Author: Mohamed Naufal Basheer <naufal11@gmail.com > */
24 /* */
25 /*****************************************************************************/
26
27 #include <sys/types.h>
28 #include <sys/mman.h>
29 #include <sys/stat.h>
30 #include <err.h>
31 #include <errno.h>
32 #include <fcntl.h>
33 #include <stdio.h>
34 #include <stdlib.h>
35 #include <unistd.h>
36
37 /*
38 * Named pipe to act as a communication channel between
39 * shell script & this process
40 */
41 #define STATUS_PIPE "status_pipe"
42
43 int flag_exit;
44 int flag_allocated;
45 unsigned long memsize;
46
47 /*
48 * process_options: process user specified options
49 */
process_options(int argc,char ** argv)50 void process_options(int argc, char **argv)
51 {
52 int c;
53 char *end;
54
55 opterr = 0;
56 while ((c = getopt(argc, argv, "pm:")) != -1) {
57 switch (c) {
58 case 'm':
59 memsize = strtoul(optarg, &end, 10);
60 if (*end != '\0')
61 errx(2, "invalid -m usage");
62 break;
63 case 'p':
64 printf("%d\n", getpagesize());
65 exit(0);
66 default:
67 errx(2, "invalid option specifed");
68 }
69 }
70
71 if (memsize <= 0)
72 errx(3, "invalid usage");
73 }
74
75 /*
76 * touch_memory: force physical memory allocation
77 */
touch_memory(char * p)78 void touch_memory(char *p)
79 {
80 int i;
81 int pagesize = getpagesize();
82
83 for (i = 0; i < memsize; i += pagesize)
84 p[i] = 0xef;
85 }
86
mem_map()87 void mem_map()
88 {
89 static char *p;
90
91 if (flag_allocated) {
92 if (munmap(p, memsize) == -1)
93 err(5, "munmap failed");
94 } else {
95 p = mmap(NULL, memsize, PROT_READ | PROT_WRITE,
96 MAP_SHARED | MAP_ANONYMOUS, 0, 0);
97 if (p == MAP_FAILED)
98 err(4, "mmap failed");
99 touch_memory(p);
100 }
101 flag_allocated = !flag_allocated;
102 }
103
104 /*
105 * done: retrieve instructions from the named pipe
106 */
action()107 char action()
108 {
109 char ch;
110 int fd;
111
112 if ((fd = open(STATUS_PIPE, O_RDONLY)) == -1)
113 err(6, "Error opening named pipe");
114
115 if (read(fd, &ch, 1) == -1)
116 err(7, "Error reading named pipe");
117
118 close(fd);
119
120 return ch;
121 }
122
main(int argc,char ** argv)123 int main(int argc, char **argv)
124 {
125 int ret;
126 char ch;
127
128 process_options(argc, argv);
129
130 ret = mkfifo(STATUS_PIPE, 0666);
131
132 if (ret == -1 && errno != EEXIST)
133 errx(1, "Error creating named pipe");
134
135 do {
136 ch = action();
137
138 if (ch == 'm')
139 mem_map();
140 } while (ch != 'x');
141
142 remove(STATUS_PIPE);
143
144 return 0;
145 }
146