-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathrestart.c
133 lines (120 loc) · 4.24 KB
/
restart.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
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <assert.h>
#include <ucontext.h>
#include <fcntl.h>
#include <sys/types.h>
#include <sys/mman.h>
#include "ckpt_sgmt.h"
/**
* A commented-out helper function which prints individual process metadata to stdout.
* @param metadata an array of ckpt_sgmt structs which contain process information which was
* captured in myckpt.dat.
* @param i int corresponding to the index corresponding the individual process in metadata[]
* to print to stdout.
*/
void print_metadata(struct ckpt_sgmt metadata[], int i) {
if (metadata[i].is_register_context == 1) {
printf("%d; metadata[%d].name: %s\n", i, i, metadata[i].name);
printf("\tmetadata[%d].is_register_context: %d\n", i, metadata[i].is_register_context);
}
else {
printf("%d; metadata[%d].name: %s\n", i, i, metadata[i].name);
printf("\tstart - end: %p - %p\n", metadata[i].start, metadata[i].end);
printf("\trwxp: %c%c%c\n",
metadata[i].rwxp[0], metadata[i].rwxp[1], metadata[i].rwxp[2]);
printf("\tis_register_context: %d\n", metadata[i].is_register_context);
printf("\tdata_size: %d\n", metadata[i].data_size);
}
}
/**
* A function which reads the myckpt.dat file, mmaps the old processes back into their approriate
* location in memory, and calls setcontext on the old ucontext_t context variable;
* accomplishes this in 5 steps:
* 1. Opens the myckpt.dat file which contains context and process metadata and data.
* 2. Decelares an array of 1000 ckpt_sgmt struct headers.
* 3. Reads the myckpt.dat file, populating the header struct;
* if the data is for ucontext_t, it is read into old_context;
* if the data is for a process, that process is mmap'd into its appropriate memory location.
* 4. Closes the open file descriptor corresponing to myckpt.dat.
* 5. Calls setcontext(&old_context) to restart the old process.
*/
void restart() {
ucontext_t old_context;
// 1. opening FILE_NAME ("myckpt.dat") file
int fd = open(FILE_NAME, O_RDONLY);
if (fd == -1) { perror("open"); }
// 2. setting up array of ckpt_sgmt "headers":
struct ckpt_sgmt metadata[1000];
// 3. reading metadata, context, and maps:
int i = 0; // keep track of metadata header index
int rc = 0; // placeholder value
while (1) {
// 3.1. reading header:
rc = read(fd, &metadata[i], sizeof(metadata[i]));
if (rc == -1) {
perror("read");
exit(1);
}
else if (rc == 0) { // i.e., EOF
break;
}
while (rc < sizeof(metadata[i])) {
rc += read(fd, &metadata[i] + rc, sizeof(metadata[i]) - rc);
}
assert(rc == sizeof(metadata[i]));
// 3.2. reading data (either ucontext_t or maps):
if (metadata[i].is_register_context == 1) { // 3.2.1; header is for ucontext_t
rc = read(fd, &old_context, sizeof(old_context));
while (rc < sizeof(old_context)) {
rc += read(fd, &old_context + rc, sizeof(old_context) - rc);
}
assert(rc == sizeof(old_context));
}
else { // 3.2.2; header is NOT for ucontext_t (i.e., a map)
void *addr = mmap(metadata[i].start, // start address for new mapping
metadata[i].data_size, // length of the mapping
PROT_READ|PROT_WRITE|PROT_EXEC, // mem map protections
MAP_FIXED|MAP_PRIVATE|MAP_ANONYMOUS, // flags
-1, // fd==-1 for MAP_ANONYMOUS
0); // offset==0 for MAP_ANONYMOUS
if (addr == MAP_FAILED) { perror("mmap"); }
rc = read(fd, (char *)addr, metadata[i].data_size);
while (rc < metadata[i].data_size) {
rc += read(fd, (char *)addr + rc, metadata[i].data_size - rc);
}
assert(rc == metadata[i].data_size);
}
//print_metadata(metadata, i);
i = i + 1;
}
// 4. closing fd:
close(fd);
// 5. setting context:
if (setcontext(&old_context) == -1) {
perror("setcontext");
exit(1);
}
}
/**
* A recursive function which is used to grow the stack with many call frames, so
* that the latest call frame will be at an address with no conflict when the
* memort segments of myckpt.dat are mmap'd back into memory.
* @param levels int representing how many recurisve calls to this function to make.
*/
void recursive(int levels) {
if (levels > 0) {
recursive(levels - 1);
}
else { // base case
restart();
}
}
/**
* Program entry point.
*/
int main() {
recursive(1000);
return 0;
}