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 default:
64 errx(2, "invalid option specified");
65 }
66 }
67
68 if (memsize <= 0)
69 errx(3, "invalid usage");
70 }
71
72 /*
73 * touch_memory: force physical memory allocation
74 */
touch_memory(char * p)75 void touch_memory(char *p)
76 {
77 int i;
78 int pagesize = getpagesize();
79
80 for (i = 0; i < (int)memsize; i += pagesize)
81 p[i] = 0xef;
82 }
83
mem_map(void)84 void mem_map(void)
85 {
86 static char *p;
87
88 if (flag_allocated) {
89 if (munmap(p, memsize) == -1)
90 err(5, "munmap failed");
91 } else {
92 p = mmap(NULL, memsize, PROT_READ | PROT_WRITE,
93 MAP_SHARED | MAP_ANONYMOUS, 0, 0);
94 if (p == MAP_FAILED)
95 err(4, "mmap failed");
96 touch_memory(p);
97 }
98 flag_allocated = !flag_allocated;
99 }
100
101 /*
102 * done: retrieve instructions from the named pipe
103 */
action(void)104 char action(void)
105 {
106 char ch;
107 int fd;
108
109 if ((fd = open(STATUS_PIPE, O_RDONLY)) == -1)
110 err(6, "Error opening named pipe");
111
112 if (read(fd, &ch, 1) == -1)
113 err(7, "Error reading named pipe");
114
115 close(fd);
116
117 return ch;
118 }
119
main(int argc,char ** argv)120 int main(int argc, char **argv)
121 {
122 int ret;
123 char ch;
124
125 process_options(argc, argv);
126
127 ret = mkfifo(STATUS_PIPE, 0666);
128
129 if (ret == -1 && errno != EEXIST)
130 errx(1, "Error creating named pipe");
131
132 do {
133 ch = action();
134
135 if (ch == 'm')
136 mem_map();
137 } while (ch != 'x');
138
139 remove(STATUS_PIPE);
140
141 return 0;
142 }
143