aboutsummaryrefslogtreecommitdiff
path: root/test/testserver.c
blob: c8052f4b2a27fc7975a7bc9e77db20730b914760 (plain)
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
#define _GNU_SOURCE
#include <stdio.h>
#include <string.h>
#include <fcntl.h>
#include <unistd.h>
#include <errno.h>
#include <signal.h>
#include <sys/socket.h>
#include <sys/un.h>
#include <sys/prctl.h>
#include "utils.h"

#define ERROR(f_, ...) fprintf(stderr, (f_), ##__VA_ARGS__)
#define UNUSED(x)      (void)(x)
#define ERROR_EXIT(f_, ...) { \
        fprintf(stderr, (f_), ##__VA_ARGS__); \
        return 1; \
    }

volatile sig_atomic_t term_caught = 0;
void onterm(int sig)
{
    printf("sigterm caught\n");
    UNUSED(sig);
    term_caught = 1;
}

int main()
{
    int result;
    int sock_fd, sock_conn;
    int nsfd;

    /* Get current namespace's file descriptor. */
    nsfd = open("/proc/self/ns/mnt", O_RDONLY);
    if (nsfd == -1)
        ERROR_EXIT("error: failed to acquire mount namespace's fd.%s\n",
                   strerror(errno));

    /* Fork. */
    pid_t ppid_before_fork = getpid();
    pid_t pid = fork();
    if (pid == -1)
        ERROR_EXIT("fork: %s\n", strerror(errno));

    if (pid == 0) {
        /* Catch SIGTERM. */
        struct sigaction sa = {0};
        sa.sa_handler = onterm;
        sigaction(SIGTERM, &sa, NULL);

        /* Ignore SIGINT. */
        signal(SIGINT, SIG_IGN);

        /* Set the child to die when parent thread dies. */
        int r = prctl(PR_SET_PDEATHSIG, SIGTERM);
        if (r == -1)
            ERROR_EXIT("prctl: %s\n", strerror(errno));
        if (getppid() != ppid_before_fork)
            ERROR_EXIT("error: parent has died already.\n");

        /* Create unix socket. */
        sock_fd = socket(AF_UNIX, SOCK_STREAM, 0);
        if (sock_fd == -1)
            ERROR_EXIT("socket: %s.\n", strerror(errno));

        struct sockaddr_un sock_addr = {0};
        sock_addr.sun_family = AF_UNIX;
        strcpy(&sock_addr.sun_path[1], "/tmp/voidnsrun-test.sock");

        if (bind(sock_fd, (struct sockaddr *)&sock_addr, sizeof(sock_addr)) == -1)
            ERROR_EXIT("bind: %s\n", strerror(errno));

        listen(sock_fd, 1);

        while (!term_caught) {
            sock_conn = accept(sock_fd, NULL, 0);
            if (sock_conn == -1) {
                ERROR("accept: %s\n", strerror(errno));
                continue;
            }
            printf("accepted\n");
            send_fd(sock_conn, nsfd);
        }
        printf("exiting\n");
    } else {
        /* This is parent. Launch a program. */
        char *argv[2] = {"/bin/sh", NULL};

        result = execvp(argv[0], (char *const *)argv);
        if (result == -1)
            ERROR_EXIT("execvp: %s\n", strerror(errno));
    }

    return 0;
}