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
|
#include <stdio.h>
#include <unistd.h>
#include <sys/wait.h>
#if defined(__linux__)
#include <sys/prctl.h>
#endif
volatile int release_child_flag = 0;
int main(int argc, char const *argv[])
{
#if defined(__linux__)
// Immediately enable any ptracer so that we can allow the stub attach
// operation to succeed. Some Linux kernels are locked down so that
// only an ancestor process can be a ptracer of a process. This disables that
// restriction. Without it, attach-related stub tests will fail.
#if defined(PR_SET_PTRACER) && defined(PR_SET_PTRACER_ANY)
// For now we execute on best effort basis. If this fails for
// some reason, so be it.
const int prctl_result = prctl(PR_SET_PTRACER, PR_SET_PTRACER_ANY, 0, 0, 0);
(void) prctl_result;
#endif
#endif
pid_t child = fork();
if (child == -1)
{
perror("fork");
return 1;
}
if (child > 0)
{ // parent
if (argc < 2)
{
fprintf(stderr, "Need pid filename.\n");
return 2;
}
// Let the test suite know the child's pid.
FILE *pid_file = fopen(argv[1], "w");
if (pid_file == NULL)
{
perror("fopen");
return 3;
}
fprintf(pid_file, "%d\n", child);
if (fclose(pid_file) == EOF)
{
perror("fclose");
return 4;
}
// And wait for the child to finish it's work.
int status = 0;
pid_t wpid = wait(&status);
if (wpid == -1)
{
perror("wait");
return 5;
}
if (wpid != child)
{
fprintf(stderr, "wait() waited for wrong child\n");
return 6;
}
if (!WIFEXITED(status) || WEXITSTATUS(status) != 0)
{
fprintf(stderr, "child did not exit correctly\n");
return 7;
}
}
else
{ // child
while (! release_child_flag) // Wait for debugger to attach
sleep(1);
printf("Child's previous process group is: %d\n", getpgid(0));
setpgid(0, 0); // Set breakpoint here
printf("Child's process group set to: %d\n", getpgid(0));
}
return 0;
}
|