blob: b4720b54c6a7409cd5a9f2c76fca84216745b5d2 [file] [log] [blame]
San Mehata1992c92010-04-07 15:21:36 -07001/*
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
17#include <stdlib.h>
18#include <errno.h>
19#include <fcntl.h>
Olivier Baillyff2c0d82010-11-17 11:45:07 -080020#include <string.h>
San Mehata1992c92010-04-07 15:21:36 -070021
22#include <sys/socket.h>
23#include <sys/stat.h>
24#include <sys/types.h>
25#include <sys/wait.h>
26
27#include <linux/netlink.h>
28#include <linux/rtnetlink.h>
29#include <linux/pkt_sched.h>
30
31#define LOG_TAG "ThrottleController"
32#include <cutils/log.h>
33
34
35#include "ThrottleController.h"
36
37static char TC_PATH[] = "/system/bin/tc";
38
JP Abgrall9e5e0ce2011-12-14 15:20:59 -080039extern "C" int system_nosh(const char *command);
San Mehat7e1f4762010-04-09 16:26:24 -070040extern "C" int ifc_init(void);
41extern "C" int ifc_up(const char *name);
42extern "C" int ifc_down(const char *name);
San Mehata1992c92010-04-07 15:21:36 -070043
44int ThrottleController::runTcCmd(const char *cmd) {
JP Abgrall11b4e9b2011-08-11 15:34:49 -070045 char *buffer;
46 size_t len = strnlen(cmd, 255);
47 int res;
San Mehata1992c92010-04-07 15:21:36 -070048
JP Abgrall11b4e9b2011-08-11 15:34:49 -070049 if (len == 255) {
Steve Block5ea0c052012-01-06 19:18:11 +000050 ALOGE("tc command too long");
JP Abgrall11b4e9b2011-08-11 15:34:49 -070051 errno = E2BIG;
52 return -1;
San Mehata1992c92010-04-07 15:21:36 -070053 }
San Mehata1992c92010-04-07 15:21:36 -070054
JP Abgrall11b4e9b2011-08-11 15:34:49 -070055 asprintf(&buffer, "%s %s", TC_PATH, cmd);
JP Abgrall9e5e0ce2011-12-14 15:20:59 -080056 res = system_nosh(buffer);
JP Abgrall11b4e9b2011-08-11 15:34:49 -070057 free(buffer);
58 return res;
San Mehata1992c92010-04-07 15:21:36 -070059}
60
61int ThrottleController::setInterfaceThrottle(const char *iface, int rxKbps, int txKbps) {
San Mehat7e1f4762010-04-09 16:26:24 -070062 char cmd[512];
63 char ifn[65];
San Mehata1992c92010-04-07 15:21:36 -070064 int rc;
65
San Mehat7e1f4762010-04-09 16:26:24 -070066 memset(ifn, 0, sizeof(ifn));
67 strncpy(ifn, iface, sizeof(ifn)-1);
68
San Mehata1992c92010-04-07 15:21:36 -070069 if (txKbps == -1) {
San Mehat7e1f4762010-04-09 16:26:24 -070070 reset(ifn);
San Mehata1992c92010-04-07 15:21:36 -070071 return 0;
72 }
73
San Mehat7e1f4762010-04-09 16:26:24 -070074 /*
75 *
76 * Target interface configuration
77 *
78 */
San Mehata1992c92010-04-07 15:21:36 -070079
San Mehat7e1f4762010-04-09 16:26:24 -070080 /*
81 * Add root qdisc for the interface
82 */
San Mehat9d8d7282010-04-30 06:19:07 -070083 sprintf(cmd, "qdisc add dev %s root handle 1: htb default 1 r2q 1000", ifn);
San Mehata1992c92010-04-07 15:21:36 -070084 if (runTcCmd(cmd)) {
Steve Block5ea0c052012-01-06 19:18:11 +000085 ALOGE("Failed to add root qdisc (%s)", strerror(errno));
San Mehat7e1f4762010-04-09 16:26:24 -070086 goto fail;
87 }
88
89 /*
90 * Add our egress throttling class
91 */
San Mehat9d8d7282010-04-30 06:19:07 -070092 sprintf(cmd, "class add dev %s parent 1: classid 1:1 htb rate %dkbit", ifn, txKbps);
San Mehat7e1f4762010-04-09 16:26:24 -070093 if (runTcCmd(cmd)) {
Steve Block5ea0c052012-01-06 19:18:11 +000094 ALOGE("Failed to add egress throttling class (%s)", strerror(errno));
San Mehat7e1f4762010-04-09 16:26:24 -070095 goto fail;
96 }
97
98 /*
San Mehat7e1f4762010-04-09 16:26:24 -070099 * Bring up the IFD device
100 */
101 ifc_init();
102 if (ifc_up("ifb0")) {
Steve Block5ea0c052012-01-06 19:18:11 +0000103 ALOGE("Failed to up ifb0 (%s)", strerror(errno));
San Mehat7e1f4762010-04-09 16:26:24 -0700104 goto fail;
105 }
106
107 /*
San Mehat9d8d7282010-04-30 06:19:07 -0700108 * Add root qdisc for IFD
109 */
110 sprintf(cmd, "qdisc add dev ifb0 root handle 1: htb default 1 r2q 1000");
111 if (runTcCmd(cmd)) {
Steve Block5ea0c052012-01-06 19:18:11 +0000112 ALOGE("Failed to add root ifb qdisc (%s)", strerror(errno));
San Mehat9d8d7282010-04-30 06:19:07 -0700113 goto fail;
114 }
115
116 /*
117 * Add our ingress throttling class
118 */
119 sprintf(cmd, "class add dev ifb0 parent 1: classid 1:1 htb rate %dkbit", rxKbps);
120 if (runTcCmd(cmd)) {
Steve Block5ea0c052012-01-06 19:18:11 +0000121 ALOGE("Failed to add ingress throttling class (%s)", strerror(errno));
San Mehat9d8d7282010-04-30 06:19:07 -0700122 goto fail;
123 }
124
125 /*
San Mehat7e1f4762010-04-09 16:26:24 -0700126 * Add ingress qdisc for pkt redirection
127 */
128 sprintf(cmd, "qdisc add dev %s ingress", ifn);
129 if (runTcCmd(cmd)) {
Steve Block5ea0c052012-01-06 19:18:11 +0000130 ALOGE("Failed to add ingress qdisc (%s)", strerror(errno));
San Mehat7e1f4762010-04-09 16:26:24 -0700131 goto fail;
132 }
133
134 /*
135 * Add filter to link <ifn> -> ifb0
136 */
San Mehat9d8d7282010-04-30 06:19:07 -0700137 sprintf(cmd, "filter add dev %s parent ffff: protocol ip prio 10 u32 match "
San Mehat7e1f4762010-04-09 16:26:24 -0700138 "u32 0 0 flowid 1:1 action mirred egress redirect dev ifb0", ifn);
139 if (runTcCmd(cmd)) {
Steve Block5ea0c052012-01-06 19:18:11 +0000140 ALOGE("Failed to add ifb filter (%s)", strerror(errno));
San Mehat7e1f4762010-04-09 16:26:24 -0700141 goto fail;
142 }
143
San Mehata1992c92010-04-07 15:21:36 -0700144 return 0;
San Mehat7e1f4762010-04-09 16:26:24 -0700145fail:
146 reset(ifn);
147 return -1;
San Mehata1992c92010-04-07 15:21:36 -0700148}
149
150void ThrottleController::reset(const char *iface) {
San Mehat7e1f4762010-04-09 16:26:24 -0700151 char cmd[128];
152
153 sprintf(cmd, "qdisc del dev %s root", iface);
San Mehata1992c92010-04-07 15:21:36 -0700154 runTcCmd(cmd);
San Mehat7e1f4762010-04-09 16:26:24 -0700155 sprintf(cmd, "qdisc del dev %s ingress", iface);
156 runTcCmd(cmd);
157
158 runTcCmd("qdisc del dev ifb0 root");
San Mehata1992c92010-04-07 15:21:36 -0700159}
160
161int ThrottleController::getInterfaceRxThrottle(const char *iface, int *rx) {
162 *rx = 0;
163 return 0;
164}
165
166int ThrottleController::getInterfaceTxThrottle(const char *iface, int *tx) {
167 *tx = 0;
168 return 0;
169}