summaryrefslogtreecommitdiff
path: root/server/ClatdController.cpp
blob: bcb01ba4575c4b29fb9f0e3f9254702d86bf0d7e (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
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
/*
 * Copyright (C) 2008 The Android Open Source Project
 *
 * Licensed under the Apache License, Version 2.0 (the "License");
 * you may not use this file except in compliance with the License.
 * You may obtain a copy of the License at
 *
 *      http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 */
#include <unistd.h>
#include <errno.h>
#include <sys/types.h>
#include <sys/wait.h>

#define LOG_TAG "ClatdController"
#include <cutils/log.h>

#include "NetdConstants.h"
#include "ClatdController.h"
#include "Fwmark.h"
#include "NetdConstants.h"
#include "NetworkController.h"

ClatdController::ClatdController(NetworkController* controller)
        : mNetCtrl(controller), mClatdPid(0) {
}

ClatdController::~ClatdController() {
}

int ClatdController::startClatd(char *interface) {
    pid_t pid;

    if(mClatdPid != 0) {
        ALOGE("clatd already running");
        errno = EBUSY;
        return -1;
    }

    if (!isIfaceName(interface)) {
        errno = ENOENT;
        return -1;
    }

    ALOGD("starting clatd");

    if ((pid = fork()) < 0) {
        ALOGE("fork failed (%s)", strerror(errno));
        return -1;
    }

    if (!pid) {
        // Pass in the interface, a netid to use for DNS lookups, and a fwmark for outgoing packets.
        unsigned netId = mNetCtrl->getNetworkForInterface(interface);
        char netIdString[UINT32_STRLEN];
        snprintf(netIdString, sizeof(netIdString), "%u", netId);

        Fwmark fwmark = {netId, true, true, PERMISSION_SYSTEM};
        char fwmarkString[UINT32_HEX_STRLEN];
        snprintf(fwmarkString, sizeof(fwmarkString), "0x%x", fwmark.intValue);

        char *args[] = {
            (char *) "/system/bin/clatd",
            (char *) "-i",
            interface,
            (char *) "-n",
            netIdString,
            (char *) "-m",
            fwmarkString,
            NULL
        };

        if (execv(args[0], args)) {
            ALOGE("execv failed (%s)", strerror(errno));
        }
        ALOGE("Should never get here!");
        _exit(0);
    } else {
        mClatdPid = pid;
        ALOGD("clatd started");
    }

    return 0;
}

int ClatdController::stopClatd() {
    if (mClatdPid == 0) {
        ALOGE("clatd already stopped");
        return -1;
    }

    ALOGD("Stopping clatd");

    kill(mClatdPid, SIGTERM);
    waitpid(mClatdPid, NULL, 0);
    mClatdPid = 0;

    ALOGD("clatd stopped");

    return 0;
}

bool ClatdController::isClatdStarted() {
    pid_t waitpid_status;
    if(mClatdPid == 0) {
        return false;
    }
    waitpid_status = waitpid(mClatdPid, NULL, WNOHANG);
    if(waitpid_status != 0) {
        mClatdPid = 0; // child exited, don't call waitpid on it again
    }
    return waitpid_status == 0; // 0 while child is running
}