/* * Copyright (C) 2016 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 "wificond/server.h" #include #include #include #include #include #include "wificond/logging_utils.h" #include "wificond/net/netlink_utils.h" #include "wificond/scanning/scan_utils.h" using android::base::WriteStringToFd; using android::binder::Status; using android::sp; using android::IBinder; using android::net::wifi::IApInterface; using android::net::wifi::IClientInterface; using android::net::wifi::IInterfaceEventCallback; using android::net::wifi::IRttClient; using android::net::wifi::IRttController; using android::wifi_system::HostapdManager; using android::wifi_system::InterfaceTool; using android::wifi_system::SupplicantManager; using std::endl; using std::placeholders::_1; using std::string; using std::stringstream; using std::unique_ptr; using std::vector; namespace android { namespace wificond { namespace { constexpr const char* kPermissionDump = "android.permission.DUMP"; } // namespace Server::Server(unique_ptr if_tool, unique_ptr supplicant_manager, unique_ptr hostapd_manager, NetlinkUtils* netlink_utils, ScanUtils* scan_utils) : if_tool_(std::move(if_tool)), supplicant_manager_(std::move(supplicant_manager)), hostapd_manager_(std::move(hostapd_manager)), netlink_utils_(netlink_utils), scan_utils_(scan_utils) { } Status Server::RegisterCallback(const sp& callback) { for (auto& it : interface_event_callbacks_) { if (IInterface::asBinder(callback) == IInterface::asBinder(it)) { LOG(WARNING) << "Ignore duplicate interface event callback registration"; return Status::ok(); } } LOG(INFO) << "New interface event callback registered"; interface_event_callbacks_.push_back(callback); return Status::ok(); } Status Server::UnregisterCallback(const sp& callback) { for (auto it = interface_event_callbacks_.begin(); it != interface_event_callbacks_.end(); it++) { if (IInterface::asBinder(callback) == IInterface::asBinder(*it)) { interface_event_callbacks_.erase(it); LOG(INFO) << "Unregister interface event callback"; return Status::ok(); } } LOG(WARNING) << "Failed to find registered interface event callback" << " to unregister"; return Status::ok(); } Status Server::registerRttClient(const sp& rtt_client, sp* out_rtt_controller) { if (rtt_controller_ == nullptr) { rtt_controller_.reset(new RttControllerImpl()); } rtt_controller_->RegisterRttClient(rtt_client); *out_rtt_controller = rtt_controller_->GetBinder(); return Status::ok(); } Status Server::unregisterRttClient(const sp& rttClient) { rtt_controller_->UnregisterRttClient(rttClient); if (rtt_controller_->GetClientCount() == 0) { rtt_controller_.reset(); } return Status::ok(); } Status Server::createApInterface(sp* created_interface) { InterfaceInfo interface; if (!SetupInterface(&interface)) { return Status::ok(); // Logging was done internally } unique_ptr ap_interface(new ApInterfaceImpl( interface.name, interface.index, netlink_utils_, if_tool_.get(), hostapd_manager_.get())); *created_interface = ap_interface->GetBinder(); ap_interfaces_.push_back(std::move(ap_interface)); BroadcastApInterfaceReady(ap_interfaces_.back()->GetBinder()); return Status::ok(); } Status Server::createClientInterface(sp* created_interface) { InterfaceInfo interface; if (!SetupInterface(&interface)) { return Status::ok(); // Logging was done internally } unique_ptr client_interface(new ClientInterfaceImpl( wiphy_index_, interface.name, interface.index, interface.mac_address, if_tool_.get(), supplicant_manager_.get(), netlink_utils_, scan_utils_)); *created_interface = client_interface->GetBinder(); client_interfaces_.push_back(std::move(client_interface)); BroadcastClientInterfaceReady(client_interfaces_.back()->GetBinder()); return Status::ok(); } Status Server::tearDownInterfaces() { for (auto& it : client_interfaces_) { BroadcastClientInterfaceTornDown(it->GetBinder()); } client_interfaces_.clear(); for (auto& it : ap_interfaces_) { BroadcastApInterfaceTornDown(it->GetBinder()); } ap_interfaces_.clear(); MarkDownAllInterfaces(); netlink_utils_->UnsubscribeRegDomainChange(wiphy_index_); return Status::ok(); } Status Server::GetClientInterfaces(vector>* out_client_interfaces) { vector> client_interfaces_binder; for (auto& it : client_interfaces_) { out_client_interfaces->push_back(asBinder(it->GetBinder())); } return binder::Status::ok(); } Status Server::GetApInterfaces(vector>* out_ap_interfaces) { vector> ap_interfaces_binder; for (auto& it : ap_interfaces_) { out_ap_interfaces->push_back(asBinder(it->GetBinder())); } return binder::Status::ok(); } status_t Server::dump(int fd, const Vector& /*args*/) { if (!PermissionCache::checkCallingPermission(String16(kPermissionDump))) { IPCThreadState* ipc = android::IPCThreadState::self(); LOG(ERROR) << "Caller (uid: " << ipc->getCallingUid() << ") is not permitted to dump wificond state"; return PERMISSION_DENIED; } stringstream ss; ss << "Current wiphy index: " << wiphy_index_ << endl; ss << "Cached interfaces list from kernel message: " << endl; for (const auto& iface : interfaces_) { ss << "Interface index: " << iface.index << ", name: " << iface.name << ", mac address: " << LoggingUtils::GetMacString(iface.mac_address) << endl; } for (const auto& iface : client_interfaces_) { iface->Dump(&ss); } for (const auto& iface : ap_interfaces_) { iface->Dump(&ss); } if (!WriteStringToFd(ss.str(), fd)) { PLOG(ERROR) << "Failed to dump state to fd " << fd; return FAILED_TRANSACTION; } return OK; } void Server::MarkDownAllInterfaces() { uint32_t wiphy_index; vector interfaces; if (netlink_utils_->GetWiphyIndex(&wiphy_index) && netlink_utils_->GetInterfaces(wiphy_index, &interfaces)) { for (InterfaceInfo& interface : interfaces) { if_tool_->SetUpState(interface.name.c_str(), false); } } } void Server::CleanUpSystemState() { supplicant_manager_->StopSupplicant(); hostapd_manager_->StopHostapd(); MarkDownAllInterfaces(); } bool Server::SetupInterface(InterfaceInfo* interface) { if (!ap_interfaces_.empty() || !client_interfaces_.empty()) { // In the future we may support multiple interfaces at once. However, // today, we support just one. LOG(ERROR) << "Cannot create AP interface when other interfaces exist"; return false; } if (!RefreshWiphyIndex()) { return false; } netlink_utils_->SubscribeRegDomainChange( wiphy_index_, std::bind(&Server::OnRegDomainChanged, this, _1)); interfaces_.clear(); if (!netlink_utils_->GetInterfaces(wiphy_index_, &interfaces_)) { LOG(ERROR) << "Failed to get interfaces info from kernel"; return false; } for (const auto& iface : interfaces_) { // Some kernel/driver uses station type for p2p interface. // In that case we can only rely on hard-coded name to exclude // p2p interface from station interfaces. if (iface.name != "p2p0") { *interface = iface; return true; } } LOG(ERROR) << "No usable interface found"; return false; } bool Server::RefreshWiphyIndex() { if (!netlink_utils_->GetWiphyIndex(&wiphy_index_)) { LOG(ERROR) << "Failed to get wiphy index"; return false; } return true; } void Server::OnRegDomainChanged(std::string& country_code) { if (country_code.empty()) { LOG(INFO) << "Regulatory domain changed"; } else { LOG(INFO) << "Regulatory domain changed to country: " << country_code; } LogSupportedBands(); } void Server::LogSupportedBands() { BandInfo band_info; ScanCapabilities scan_capabilities; WiphyFeatures wiphy_features; netlink_utils_->GetWiphyInfo(wiphy_index_, &band_info, &scan_capabilities, &wiphy_features); stringstream ss; for (unsigned int i = 0; i < band_info.band_2g.size(); i++) { ss << " " << band_info.band_2g[i]; } LOG(INFO) << "2.4Ghz frequencies:"<< ss.str(); ss.str(""); for (unsigned int i = 0; i < band_info.band_5g.size(); i++) { ss << " " << band_info.band_5g[i]; } LOG(INFO) << "5Ghz non-DFS frequencies:"<< ss.str(); ss.str(""); for (unsigned int i = 0; i < band_info.band_dfs.size(); i++) { ss << " " << band_info.band_dfs[i]; } LOG(INFO) << "5Ghz DFS frequencies:"<< ss.str(); } void Server::BroadcastClientInterfaceReady( sp network_interface) { for (auto& it : interface_event_callbacks_) { it->OnClientInterfaceReady(network_interface); } } void Server::BroadcastApInterfaceReady( sp network_interface) { for (auto& it : interface_event_callbacks_) { it->OnApInterfaceReady(network_interface); } } void Server::BroadcastClientInterfaceTornDown( sp network_interface) { for (auto& it : interface_event_callbacks_) { it->OnClientTorndownEvent(network_interface); } } void Server::BroadcastApInterfaceTornDown( sp network_interface) { for (auto& it : interface_event_callbacks_) { it->OnApTorndownEvent(network_interface); } } } // namespace wificond } // namespace android