blob: c19e299aa633f476e3ea1890606bcdeb859850e5 [file] [log] [blame]
Daniel Drown0da73fc2012-06-20 16:51:39 -05001/*
2 * Copyright (C) 2008 The Android Open Source Project
3 *
4 * Licensed under the Apache License, Version 2.0 (the "License");
5 * you may not use this file except in compliance with the License.
6 * You may obtain a copy of the License at
7 *
8 * http://www.apache.org/licenses/LICENSE-2.0
9 *
10 * Unless required by applicable law or agreed to in writing, software
11 * distributed under the License is distributed on an "AS IS" BASIS,
12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 * See the License for the specific language governing permissions and
14 * limitations under the License.
15 */
16#include <unistd.h>
17#include <errno.h>
18#include <sys/types.h>
19#include <sys/wait.h>
20
21#define LOG_TAG "ClatdController"
22#include <cutils/log.h>
23
JP Abgrall69261cb2014-06-19 18:35:24 -070024#include "NetdConstants.h"
Daniel Drown0da73fc2012-06-20 16:51:39 -050025#include "ClatdController.h"
Lorenzo Colitti45d3dd02014-06-09 14:09:20 +090026#include "Fwmark.h"
Paul Jensen84c1d032014-05-30 13:29:41 -040027#include "NetdConstants.h"
28#include "NetworkController.h"
Daniel Drown0da73fc2012-06-20 16:51:39 -050029
Paul Jensen84c1d032014-05-30 13:29:41 -040030ClatdController::ClatdController(NetworkController* controller)
31 : mNetCtrl(controller), mClatdPid(0) {
Daniel Drown0da73fc2012-06-20 16:51:39 -050032}
33
34ClatdController::~ClatdController() {
35}
36
37int ClatdController::startClatd(char *interface) {
38 pid_t pid;
39
40 if(mClatdPid != 0) {
41 ALOGE("clatd already running");
42 errno = EBUSY;
43 return -1;
44 }
45
JP Abgrall69261cb2014-06-19 18:35:24 -070046 if (!isIfaceName(interface)) {
47 errno = ENOENT;
48 return -1;
49 }
50
Daniel Drown0da73fc2012-06-20 16:51:39 -050051 ALOGD("starting clatd");
52
53 if ((pid = fork()) < 0) {
54 ALOGE("fork failed (%s)", strerror(errno));
55 return -1;
56 }
57
58 if (!pid) {
Lorenzo Colitti45d3dd02014-06-09 14:09:20 +090059 // Pass in the interface, a netid to use for DNS lookups, and a fwmark for outgoing packets.
Sreeram Ramachandrane09b20a2014-07-05 17:15:14 -070060 unsigned netId = mNetCtrl->getNetworkForInterface(interface);
Lorenzo Colitti45d3dd02014-06-09 14:09:20 +090061 char netIdString[UINT32_STRLEN];
62 snprintf(netIdString, sizeof(netIdString), "%u", netId);
63
Sreeram Ramachandran335f2932014-07-11 16:01:33 -070064 Fwmark fwmark;
65
66 fwmark.netId = netId;
67 fwmark.explicitlySelected = true;
68 fwmark.protectedFromVpn = true;
69 fwmark.permission = PERMISSION_SYSTEM;
70
Lorenzo Colitti45d3dd02014-06-09 14:09:20 +090071 char fwmarkString[UINT32_HEX_STRLEN];
72 snprintf(fwmarkString, sizeof(fwmarkString), "0x%x", fwmark.intValue);
73
Paul Jensen84c1d032014-05-30 13:29:41 -040074 char *args[] = {
Lorenzo Colitti45d3dd02014-06-09 14:09:20 +090075 (char *) "/system/bin/clatd",
76 (char *) "-i",
Paul Jensen84c1d032014-05-30 13:29:41 -040077 interface,
Lorenzo Colitti45d3dd02014-06-09 14:09:20 +090078 (char *) "-n",
79 netIdString,
80 (char *) "-m",
81 fwmarkString,
Paul Jensen84c1d032014-05-30 13:29:41 -040082 NULL
83 };
Daniel Drown0da73fc2012-06-20 16:51:39 -050084
85 if (execv(args[0], args)) {
86 ALOGE("execv failed (%s)", strerror(errno));
87 }
88 ALOGE("Should never get here!");
Daniel Drown0da73fc2012-06-20 16:51:39 -050089 _exit(0);
90 } else {
91 mClatdPid = pid;
92 ALOGD("clatd started");
93 }
94
95 return 0;
96}
97
98int ClatdController::stopClatd() {
99 if (mClatdPid == 0) {
100 ALOGE("clatd already stopped");
101 return -1;
102 }
103
104 ALOGD("Stopping clatd");
105
106 kill(mClatdPid, SIGTERM);
107 waitpid(mClatdPid, NULL, 0);
108 mClatdPid = 0;
109
110 ALOGD("clatd stopped");
111
112 return 0;
113}
114
115bool ClatdController::isClatdStarted() {
116 pid_t waitpid_status;
117 if(mClatdPid == 0) {
118 return false;
119 }
120 waitpid_status = waitpid(mClatdPid, NULL, WNOHANG);
121 if(waitpid_status != 0) {
122 mClatdPid = 0; // child exited, don't call waitpid on it again
123 }
124 return waitpid_status == 0; // 0 while child is running
125}