1d18304287dbabc7835be771400b85d4ae8b63de6San Mehat/*
2d18304287dbabc7835be771400b85d4ae8b63de6San Mehat * Copyright (C) 2008 The Android Open Source Project
3d18304287dbabc7835be771400b85d4ae8b63de6San Mehat *
4d18304287dbabc7835be771400b85d4ae8b63de6San Mehat * Licensed under the Apache License, Version 2.0 (the "License");
5d18304287dbabc7835be771400b85d4ae8b63de6San Mehat * you may not use this file except in compliance with the License.
6d18304287dbabc7835be771400b85d4ae8b63de6San Mehat * You may obtain a copy of the License at
7d18304287dbabc7835be771400b85d4ae8b63de6San Mehat *
8d18304287dbabc7835be771400b85d4ae8b63de6San Mehat *      http://www.apache.org/licenses/LICENSE-2.0
9d18304287dbabc7835be771400b85d4ae8b63de6San Mehat *
10d18304287dbabc7835be771400b85d4ae8b63de6San Mehat * Unless required by applicable law or agreed to in writing, software
11d18304287dbabc7835be771400b85d4ae8b63de6San Mehat * distributed under the License is distributed on an "AS IS" BASIS,
12d18304287dbabc7835be771400b85d4ae8b63de6San Mehat * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13d18304287dbabc7835be771400b85d4ae8b63de6San Mehat * See the License for the specific language governing permissions and
14d18304287dbabc7835be771400b85d4ae8b63de6San Mehat * limitations under the License.
15d18304287dbabc7835be771400b85d4ae8b63de6San Mehat */
16d18304287dbabc7835be771400b85d4ae8b63de6San Mehat
17d18304287dbabc7835be771400b85d4ae8b63de6San Mehat#include <stdio.h>
18d18304287dbabc7835be771400b85d4ae8b63de6San Mehat#include <stdlib.h>
19ff2c0d8c13457e43f0d4bf06d3177271aac104c1Olivier Bailly#include <string.h>
20d18304287dbabc7835be771400b85d4ae8b63de6San Mehat#include <errno.h>
21d18304287dbabc7835be771400b85d4ae8b63de6San Mehat
22d18304287dbabc7835be771400b85d4ae8b63de6San Mehat#define LOG_TAG "Netd"
23d18304287dbabc7835be771400b85d4ae8b63de6San Mehat
24d18304287dbabc7835be771400b85d4ae8b63de6San Mehat#include <cutils/log.h>
25d18304287dbabc7835be771400b85d4ae8b63de6San Mehat
26d18304287dbabc7835be771400b85d4ae8b63de6San Mehat#include <sysutils/NetlinkEvent.h>
27d18304287dbabc7835be771400b85d4ae8b63de6San Mehat#include "NetlinkHandler.h"
2867c5753274ee828b56f96b5e9cdae64c8562ad52Robert Greenwalt#include "NetlinkManager.h"
2967c5753274ee828b56f96b5e9cdae64c8562ad52Robert Greenwalt#include "ResponseCode.h"
30d18304287dbabc7835be771400b85d4ae8b63de6San Mehat
31564df4efc87f3d04c3570836d38134fd949c4cafMike J. ChenNetlinkHandler::NetlinkHandler(NetlinkManager *nm, int listenerSocket,
32564df4efc87f3d04c3570836d38134fd949c4cafMike J. Chen                               int format) :
33564df4efc87f3d04c3570836d38134fd949c4cafMike J. Chen                        NetlinkListener(listenerSocket, format) {
3467c5753274ee828b56f96b5e9cdae64c8562ad52Robert Greenwalt    mNm = nm;
35d18304287dbabc7835be771400b85d4ae8b63de6San Mehat}
36d18304287dbabc7835be771400b85d4ae8b63de6San Mehat
37d18304287dbabc7835be771400b85d4ae8b63de6San MehatNetlinkHandler::~NetlinkHandler() {
38d18304287dbabc7835be771400b85d4ae8b63de6San Mehat}
39d18304287dbabc7835be771400b85d4ae8b63de6San Mehat
40d18304287dbabc7835be771400b85d4ae8b63de6San Mehatint NetlinkHandler::start() {
41d18304287dbabc7835be771400b85d4ae8b63de6San Mehat    return this->startListener();
42d18304287dbabc7835be771400b85d4ae8b63de6San Mehat}
43d18304287dbabc7835be771400b85d4ae8b63de6San Mehat
44d18304287dbabc7835be771400b85d4ae8b63de6San Mehatint NetlinkHandler::stop() {
45d18304287dbabc7835be771400b85d4ae8b63de6San Mehat    return this->stopListener();
46d18304287dbabc7835be771400b85d4ae8b63de6San Mehat}
47d18304287dbabc7835be771400b85d4ae8b63de6San Mehat
48d18304287dbabc7835be771400b85d4ae8b63de6San Mehatvoid NetlinkHandler::onEvent(NetlinkEvent *evt) {
49d18304287dbabc7835be771400b85d4ae8b63de6San Mehat    const char *subsys = evt->getSubsystem();
50d18304287dbabc7835be771400b85d4ae8b63de6San Mehat    if (!subsys) {
510e76b761a1514d5182675dd7b7d33725f62d6bc5Steve Block        ALOGW("No subsystem found in netlink event");
52d18304287dbabc7835be771400b85d4ae8b63de6San Mehat        return;
53d18304287dbabc7835be771400b85d4ae8b63de6San Mehat    }
54564df4efc87f3d04c3570836d38134fd949c4cafMike J. Chen
5567c5753274ee828b56f96b5e9cdae64c8562ad52Robert Greenwalt    if (!strcmp(subsys, "net")) {
5667c5753274ee828b56f96b5e9cdae64c8562ad52Robert Greenwalt        int action = evt->getAction();
57564df4efc87f3d04c3570836d38134fd949c4cafMike J. Chen        const char *iface = evt->findParam("INTERFACE");
58564df4efc87f3d04c3570836d38134fd949c4cafMike J. Chen
5967c5753274ee828b56f96b5e9cdae64c8562ad52Robert Greenwalt        if (action == evt->NlActionAdd) {
6067c5753274ee828b56f96b5e9cdae64c8562ad52Robert Greenwalt            notifyInterfaceAdded(iface);
6167c5753274ee828b56f96b5e9cdae64c8562ad52Robert Greenwalt        } else if (action == evt->NlActionRemove) {
6267c5753274ee828b56f96b5e9cdae64c8562ad52Robert Greenwalt            notifyInterfaceRemoved(iface);
6367c5753274ee828b56f96b5e9cdae64c8562ad52Robert Greenwalt        } else if (action == evt->NlActionChange) {
6467c5753274ee828b56f96b5e9cdae64c8562ad52Robert Greenwalt            evt->dump();
6567c5753274ee828b56f96b5e9cdae64c8562ad52Robert Greenwalt            notifyInterfaceChanged("nana", true);
66564df4efc87f3d04c3570836d38134fd949c4cafMike J. Chen        } else if (action == evt->NlActionLinkUp) {
67564df4efc87f3d04c3570836d38134fd949c4cafMike J. Chen            notifyInterfaceLinkChanged(iface, true);
68564df4efc87f3d04c3570836d38134fd949c4cafMike J. Chen        } else if (action == evt->NlActionLinkDown) {
69564df4efc87f3d04c3570836d38134fd949c4cafMike J. Chen            notifyInterfaceLinkChanged(iface, false);
7067c5753274ee828b56f96b5e9cdae64c8562ad52Robert Greenwalt        }
71e07effe6f8e9340dbee9428b29672adfb647c413JP Abgrall
72e0ebc46c0aa38ce4f35bd3b60c0fcb9204d4c35eJP Abgrall    } else if (!strcmp(subsys, "qlog")) {
73e0ebc46c0aa38ce4f35bd3b60c0fcb9204d4c35eJP Abgrall        const char *alertName = evt->findParam("ALERT_NAME");
74e0ebc46c0aa38ce4f35bd3b60c0fcb9204d4c35eJP Abgrall        const char *iface = evt->findParam("INTERFACE");
75e0ebc46c0aa38ce4f35bd3b60c0fcb9204d4c35eJP Abgrall        notifyQuotaLimitReached(alertName, iface);
76e07effe6f8e9340dbee9428b29672adfb647c413JP Abgrall
77e07effe6f8e9340dbee9428b29672adfb647c413JP Abgrall    } else if (!strcmp(subsys, "xt_idletimer")) {
786337b88ce4438d224819e9b381ddaf2873bbfddaAshish Sharma        int action = evt->getAction();
7998f65d32b1530b4da6050e38d52f955710577efbHaoyu Bai        const char *label = evt->findParam("LABEL");
80e07effe6f8e9340dbee9428b29672adfb647c413JP Abgrall        const char *state = evt->findParam("STATE");
8198f65d32b1530b4da6050e38d52f955710577efbHaoyu Bai        // if no LABEL, use INTERFACE instead
8298f65d32b1530b4da6050e38d52f955710577efbHaoyu Bai        if (label == NULL) {
8398f65d32b1530b4da6050e38d52f955710577efbHaoyu Bai            label = evt->findParam("INTERFACE");
8498f65d32b1530b4da6050e38d52f955710577efbHaoyu Bai        }
85e07effe6f8e9340dbee9428b29672adfb647c413JP Abgrall        if (state)
8698f65d32b1530b4da6050e38d52f955710577efbHaoyu Bai            notifyInterfaceClassActivity(label, !strcmp("active", state));
87e0ebc46c0aa38ce4f35bd3b60c0fcb9204d4c35eJP Abgrall
8840baed83f89e8d8f834cdfa94d0615076463c04bJP Abgrall#if !LOG_NDEBUG
8940baed83f89e8d8f834cdfa94d0615076463c04bJP Abgrall    } else if (strcmp(subsys, "platform") && strcmp(subsys, "backlight")) {
9040baed83f89e8d8f834cdfa94d0615076463c04bJP Abgrall        /* It is not a VSYNC or a backlight event */
9140baed83f89e8d8f834cdfa94d0615076463c04bJP Abgrall        ALOGV("unexpected event from subsystem %s", subsys);
9240baed83f89e8d8f834cdfa94d0615076463c04bJP Abgrall#endif
936337b88ce4438d224819e9b381ddaf2873bbfddaAshish Sharma    }
94d18304287dbabc7835be771400b85d4ae8b63de6San Mehat}
9567c5753274ee828b56f96b5e9cdae64c8562ad52Robert Greenwalt
9667c5753274ee828b56f96b5e9cdae64c8562ad52Robert Greenwaltvoid NetlinkHandler::notifyInterfaceAdded(const char *name) {
9767c5753274ee828b56f96b5e9cdae64c8562ad52Robert Greenwalt    char msg[255];
9867c5753274ee828b56f96b5e9cdae64c8562ad52Robert Greenwalt    snprintf(msg, sizeof(msg), "Iface added %s", name);
9967c5753274ee828b56f96b5e9cdae64c8562ad52Robert Greenwalt
10067c5753274ee828b56f96b5e9cdae64c8562ad52Robert Greenwalt    mNm->getBroadcaster()->sendBroadcast(ResponseCode::InterfaceChange,
10167c5753274ee828b56f96b5e9cdae64c8562ad52Robert Greenwalt            msg, false);
10267c5753274ee828b56f96b5e9cdae64c8562ad52Robert Greenwalt}
10367c5753274ee828b56f96b5e9cdae64c8562ad52Robert Greenwalt
10467c5753274ee828b56f96b5e9cdae64c8562ad52Robert Greenwaltvoid NetlinkHandler::notifyInterfaceRemoved(const char *name) {
10567c5753274ee828b56f96b5e9cdae64c8562ad52Robert Greenwalt    char msg[255];
10667c5753274ee828b56f96b5e9cdae64c8562ad52Robert Greenwalt    snprintf(msg, sizeof(msg), "Iface removed %s", name);
10767c5753274ee828b56f96b5e9cdae64c8562ad52Robert Greenwalt
10867c5753274ee828b56f96b5e9cdae64c8562ad52Robert Greenwalt    mNm->getBroadcaster()->sendBroadcast(ResponseCode::InterfaceChange,
10967c5753274ee828b56f96b5e9cdae64c8562ad52Robert Greenwalt            msg, false);
11067c5753274ee828b56f96b5e9cdae64c8562ad52Robert Greenwalt}
11167c5753274ee828b56f96b5e9cdae64c8562ad52Robert Greenwalt
11267c5753274ee828b56f96b5e9cdae64c8562ad52Robert Greenwaltvoid NetlinkHandler::notifyInterfaceChanged(const char *name, bool isUp) {
11367c5753274ee828b56f96b5e9cdae64c8562ad52Robert Greenwalt    char msg[255];
114564df4efc87f3d04c3570836d38134fd949c4cafMike J. Chen    snprintf(msg, sizeof(msg), "Iface changed %s %s", name,
115564df4efc87f3d04c3570836d38134fd949c4cafMike J. Chen             (isUp ? "up" : "down"));
116564df4efc87f3d04c3570836d38134fd949c4cafMike J. Chen
117564df4efc87f3d04c3570836d38134fd949c4cafMike J. Chen    mNm->getBroadcaster()->sendBroadcast(ResponseCode::InterfaceChange,
118564df4efc87f3d04c3570836d38134fd949c4cafMike J. Chen            msg, false);
119564df4efc87f3d04c3570836d38134fd949c4cafMike J. Chen}
120564df4efc87f3d04c3570836d38134fd949c4cafMike J. Chen
121564df4efc87f3d04c3570836d38134fd949c4cafMike J. Chenvoid NetlinkHandler::notifyInterfaceLinkChanged(const char *name, bool isUp) {
122564df4efc87f3d04c3570836d38134fd949c4cafMike J. Chen    char msg[255];
123a583f8be6126d3116311e96c6fd8bafcbaf2b431Mike J. Chen    snprintf(msg, sizeof(msg), "Iface linkstate %s %s", name,
124564df4efc87f3d04c3570836d38134fd949c4cafMike J. Chen             (isUp ? "up" : "down"));
12567c5753274ee828b56f96b5e9cdae64c8562ad52Robert Greenwalt
12667c5753274ee828b56f96b5e9cdae64c8562ad52Robert Greenwalt    mNm->getBroadcaster()->sendBroadcast(ResponseCode::InterfaceChange,
12767c5753274ee828b56f96b5e9cdae64c8562ad52Robert Greenwalt            msg, false);
12867c5753274ee828b56f96b5e9cdae64c8562ad52Robert Greenwalt}
129e0ebc46c0aa38ce4f35bd3b60c0fcb9204d4c35eJP Abgrall
130e0ebc46c0aa38ce4f35bd3b60c0fcb9204d4c35eJP Abgrallvoid NetlinkHandler::notifyQuotaLimitReached(const char *name, const char *iface) {
131e0ebc46c0aa38ce4f35bd3b60c0fcb9204d4c35eJP Abgrall    char msg[255];
132e0ebc46c0aa38ce4f35bd3b60c0fcb9204d4c35eJP Abgrall    snprintf(msg, sizeof(msg), "limit alert %s %s", name, iface);
133e0ebc46c0aa38ce4f35bd3b60c0fcb9204d4c35eJP Abgrall
134e0ebc46c0aa38ce4f35bd3b60c0fcb9204d4c35eJP Abgrall    mNm->getBroadcaster()->sendBroadcast(ResponseCode::BandwidthControl,
135e0ebc46c0aa38ce4f35bd3b60c0fcb9204d4c35eJP Abgrall            msg, false);
136e0ebc46c0aa38ce4f35bd3b60c0fcb9204d4c35eJP Abgrall}
1376337b88ce4438d224819e9b381ddaf2873bbfddaAshish Sharma
13898f65d32b1530b4da6050e38d52f955710577efbHaoyu Baivoid NetlinkHandler::notifyInterfaceClassActivity(const char *name,
13998f65d32b1530b4da6050e38d52f955710577efbHaoyu Bai                                                  bool isActive) {
1406337b88ce4438d224819e9b381ddaf2873bbfddaAshish Sharma    char msg[255];
1416337b88ce4438d224819e9b381ddaf2873bbfddaAshish Sharma
142db28cbecc5cddae4dff740a299d311f18ddea523Haoyu Bai    snprintf(msg, sizeof(msg), "IfaceClass %s %s",
143db28cbecc5cddae4dff740a299d311f18ddea523Haoyu Bai             isActive ? "active" : "idle", name);
144e07effe6f8e9340dbee9428b29672adfb647c413JP Abgrall    ALOGV("Broadcasting interface activity msg: %s", msg);
14598f65d32b1530b4da6050e38d52f955710577efbHaoyu Bai    mNm->getBroadcaster()->sendBroadcast(
146db28cbecc5cddae4dff740a299d311f18ddea523Haoyu Bai        ResponseCode::InterfaceClassActivity, msg, false);
1476337b88ce4438d224819e9b381ddaf2873bbfddaAshish Sharma}
148