| /* |
| * 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 <stdio.h> |
| #include <stdlib.h> |
| #include <string.h> |
| #include <errno.h> |
| |
| #define LOG_TAG "Netd" |
| |
| #include <cutils/log.h> |
| |
| #include <sysutils/NetlinkEvent.h> |
| #include "NetlinkHandler.h" |
| #include "NetlinkManager.h" |
| #include "ResponseCode.h" |
| |
| NetlinkHandler::NetlinkHandler(NetlinkManager *nm, int listenerSocket, |
| int format) : |
| NetlinkListener(listenerSocket, format) { |
| mNm = nm; |
| } |
| |
| NetlinkHandler::~NetlinkHandler() { |
| } |
| |
| int NetlinkHandler::start() { |
| return this->startListener(); |
| } |
| |
| int NetlinkHandler::stop() { |
| return this->stopListener(); |
| } |
| |
| void NetlinkHandler::onEvent(NetlinkEvent *evt) { |
| const char *subsys = evt->getSubsystem(); |
| if (!subsys) { |
| ALOGW("No subsystem found in netlink event"); |
| return; |
| } |
| |
| if (!strcmp(subsys, "net")) { |
| int action = evt->getAction(); |
| const char *iface = evt->findParam("INTERFACE"); |
| |
| if (action == evt->NlActionAdd) { |
| notifyInterfaceAdded(iface); |
| } else if (action == evt->NlActionRemove) { |
| notifyInterfaceRemoved(iface); |
| } else if (action == evt->NlActionChange) { |
| evt->dump(); |
| notifyInterfaceChanged("nana", true); |
| } else if (action == evt->NlActionLinkUp) { |
| notifyInterfaceLinkChanged(iface, true); |
| } else if (action == evt->NlActionLinkDown) { |
| notifyInterfaceLinkChanged(iface, false); |
| } |
| |
| } else if (!strcmp(subsys, "qlog")) { |
| const char *alertName = evt->findParam("ALERT_NAME"); |
| const char *iface = evt->findParam("INTERFACE"); |
| notifyQuotaLimitReached(alertName, iface); |
| |
| } else if (!strcmp(subsys, "xt_idletimer")) { |
| int action = evt->getAction(); |
| const char *label = evt->findParam("LABEL"); |
| const char *state = evt->findParam("STATE"); |
| // if no LABEL, use INTERFACE instead |
| if (label == NULL) { |
| label = evt->findParam("INTERFACE"); |
| } |
| if (state) |
| notifyInterfaceClassActivity(label, !strcmp("active", state)); |
| |
| #if !LOG_NDEBUG |
| } else if (strcmp(subsys, "platform") && strcmp(subsys, "backlight")) { |
| /* It is not a VSYNC or a backlight event */ |
| ALOGV("unexpected event from subsystem %s", subsys); |
| #endif |
| } |
| } |
| |
| void NetlinkHandler::notifyInterfaceAdded(const char *name) { |
| char msg[255]; |
| snprintf(msg, sizeof(msg), "Iface added %s", name); |
| |
| mNm->getBroadcaster()->sendBroadcast(ResponseCode::InterfaceChange, |
| msg, false); |
| } |
| |
| void NetlinkHandler::notifyInterfaceRemoved(const char *name) { |
| char msg[255]; |
| snprintf(msg, sizeof(msg), "Iface removed %s", name); |
| |
| mNm->getBroadcaster()->sendBroadcast(ResponseCode::InterfaceChange, |
| msg, false); |
| } |
| |
| void NetlinkHandler::notifyInterfaceChanged(const char *name, bool isUp) { |
| char msg[255]; |
| snprintf(msg, sizeof(msg), "Iface changed %s %s", name, |
| (isUp ? "up" : "down")); |
| |
| mNm->getBroadcaster()->sendBroadcast(ResponseCode::InterfaceChange, |
| msg, false); |
| } |
| |
| void NetlinkHandler::notifyInterfaceLinkChanged(const char *name, bool isUp) { |
| char msg[255]; |
| snprintf(msg, sizeof(msg), "Iface linkstate %s %s", name, |
| (isUp ? "up" : "down")); |
| |
| mNm->getBroadcaster()->sendBroadcast(ResponseCode::InterfaceChange, |
| msg, false); |
| } |
| |
| void NetlinkHandler::notifyQuotaLimitReached(const char *name, const char *iface) { |
| char msg[255]; |
| snprintf(msg, sizeof(msg), "limit alert %s %s", name, iface); |
| |
| mNm->getBroadcaster()->sendBroadcast(ResponseCode::BandwidthControl, |
| msg, false); |
| } |
| |
| void NetlinkHandler::notifyInterfaceClassActivity(const char *name, |
| bool isActive) { |
| char msg[255]; |
| |
| snprintf(msg, sizeof(msg), "IfaceClass %s %s", |
| isActive ? "active" : "idle", name); |
| ALOGV("Broadcasting interface activity msg: %s", msg); |
| mNm->getBroadcaster()->sendBroadcast( |
| ResponseCode::InterfaceClassActivity, msg, false); |
| } |