-
Notifications
You must be signed in to change notification settings - Fork 6
/
Copy pathproc.c
141 lines (119 loc) · 4.41 KB
/
proc.c
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
#include <stdbool.h>
#include <search.h>
#include <unistd.h>
#include <glib.h>
#include <sched.h>
#include <errno.h>
#include <stdlib.h>
#include <sys/ipc.h>
#include <sys/shm.h>
#include <sys/types.h>
#include <sys/wait.h>
#include <sys/stat.h>
#include <sys/mman.h>
#include <sys/syscall.h>
#include "sysfuzz.h"
#include "typelib.h"
#include "iknowthis.h"
// FIXME: Why is this protected by #ifdef _KERNEL in freebsd? How do userspace
// programs do it?
#if defined(__FreeBSD__)
struct shminfo {
u_long shmmax; /* max shared memory segment size (bytes) */
u_long shmmin; /* max shared memory segment size (bytes) */
u_long shmmni; /* max number of shared memory identifiers */
u_long shmseg; /* max shared memory segments per process */
u_long shmall; /* max amount of shared memory (pages) */
};
#endif
// We need to cap the number of processes created to avoid fork bombing the
// system. The obvious solution is setrlimit(RLIM_NPROC), but this is uid wide
// and will intefere with the operation of the rest of the system. LWPs also
// count against this limit, which would interfere with fuzzing.
//
// We need a kernel supported listener count, that we can check. I considered
// sockets, files, semaphores, but I think the most reliable is shmid attach
// counts.
//
// Note that zombies do _not_ count against this limit. Zombies are cleaned up
// by the resource management code when they are evicted from the list.
// The shmid for this process.
static gint shmid;
// Remove any old shared memory segments left over from previous runs.
void clear_shared_segments(uid_t owner)
{
struct shminfo shminfo;
struct shmid_ds shmds;
#if defined(__FreeBSD__) || defined(__OpenBSD__)
# warning FIXME how do you do this on BSD?
return;
#else
gint numshm = shmctl(0, SHM_INFO, &shminfo);
// I don't think these can fail.
g_assert_cmpint(numshm, >=, 0);
// Remove all existing ipc stuff that may have been left behind from previous runs.
while (numshm--) {
gint shmid = shmctl(numshm, IPC_STAT, &shmds);
if (shmds.shm_perm.uid == owner) {
shmctl(shmid, IPC_RMID, NULL);
}
}
// TODO: msgctl, etc
return;
#endif
}
void __constructor create_process_shmid(void)
{
struct shmid_ds shmds;
// I should only be called once.
g_assert_cmpint(shmid, ==, 0);
#ifndef __linux__
# warning FIXME
return;
#endif
// Create a shmid, used to track process creations.
// ftok() does a stat(), and uses the inode number combined with the
// proj_id, so I'll use the process id.
if ((shmid = shmget(ftok("/proc/self/exe", getpid()), PAGE_SIZE, IPC_CREAT | 0666)) == -1) {
// I cannot safely continue, or I might take the system down.
g_critical("unable to create a shared segment id to track processes, %s", custom_strerror_wrapper(errno));
abort();
}
// Attach to this segment.
if (shmat(shmid, NULL, 0) == MAP_FAILED) {
// This is probably not good.
g_critical("there was an error attaching to shmid %#x, %s.", shmid, custom_strerror_wrapper(errno));
abort();
}
// Mark the shmid for deletion so that it's destroyed on exit.
// This isn't really essential, it just prevents leaving a mess behind.
shmctl(shmid, IPC_RMID, NULL);
// Quick sanity check to ensure the nattach count is working.
g_assert_cmpint(shmctl(shmid, IPC_STAT, &shmds), !=, -1);
g_assert_cmpint(shmds.shm_nattch, ==, 1);
return;
}
// Increment the spawned process count for this fuzzing session, and return the
// current number of processes.
guint increment_process_count(void)
{
// Attach to the process shm segment to increment the attach count.
if (shmat(shmid, NULL, SHM_RDONLY) == MAP_FAILED) {
g_critical("there was an error attaching to shmid %#x, %s.", shmid, custom_strerror_wrapper(errno));
abort();
}
// Return to caller so that they can verify this is within bounds.
return get_process_count();
}
// Get the current number of processes, but don't increment it.
guint get_process_count(void)
{
struct shmid_ds shmds;
// Stat the id to find the current count.
if (shmctl(shmid, IPC_STAT, &shmds) == -1) {
g_critical("there was an error stating the process counting shmid, %s", custom_strerror_wrapper(errno));
abort();
}
// Return to caller so that they can verify this is within bounds.
return shmds.shm_nattch;
}