• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /*
2  * Copyright (C) 2019 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 <net/if.h>
18 #include <cstddef>
19 #include <iostream>
20 #include <limits>
21 #include <random>
22 
23 #include <android-base/logging.h>
24 #include <android-base/macros.h>
25 #include <private/android_filesystem_config.h>
26 
27 #undef NAN
28 #include "wifi_iface_util.h"
29 
30 namespace {
31 // Constants to set the local bit & clear the multicast bit.
32 constexpr uint8_t kMacAddressMulticastMask = 0x01;
33 constexpr uint8_t kMacAddressLocallyAssignedMask = 0x02;
34 }  // namespace
35 
36 namespace android {
37 namespace hardware {
38 namespace wifi {
39 namespace V1_6 {
40 namespace implementation {
41 namespace iface_util {
42 
WifiIfaceUtil(const std::weak_ptr<wifi_system::InterfaceTool> iface_tool,const std::weak_ptr<legacy_hal::WifiLegacyHal> legacy_hal)43 WifiIfaceUtil::WifiIfaceUtil(const std::weak_ptr<wifi_system::InterfaceTool> iface_tool,
44                              const std::weak_ptr<legacy_hal::WifiLegacyHal> legacy_hal)
45     : iface_tool_(iface_tool),
46       legacy_hal_(legacy_hal),
47       random_mac_address_(nullptr),
48       event_handlers_map_() {}
49 
getFactoryMacAddress(const std::string & iface_name)50 std::array<uint8_t, 6> WifiIfaceUtil::getFactoryMacAddress(const std::string& iface_name) {
51     return iface_tool_.lock()->GetFactoryMacAddress(iface_name.c_str());
52 }
53 
setMacAddress(const std::string & iface_name,const std::array<uint8_t,6> & mac)54 bool WifiIfaceUtil::setMacAddress(const std::string& iface_name,
55                                   const std::array<uint8_t, 6>& mac) {
56 #ifndef WIFI_AVOID_IFACE_RESET_MAC_CHANGE
57     legacy_hal::wifi_error legacy_status;
58     uint64_t legacy_feature_set;
59     std::tie(legacy_status, legacy_feature_set) =
60             legacy_hal_.lock()->getSupportedFeatureSet(iface_name);
61 
62     if (!(legacy_feature_set & WIFI_FEATURE_DYNAMIC_SET_MAC) &&
63         !iface_tool_.lock()->SetUpState(iface_name.c_str(), false)) {
64         LOG(ERROR) << "SetUpState(false) failed.";
65         return false;
66     }
67 #endif
68     bool success = iface_tool_.lock()->SetMacAddress(iface_name.c_str(), mac);
69 #ifndef WIFI_AVOID_IFACE_RESET_MAC_CHANGE
70     if (!(legacy_feature_set & WIFI_FEATURE_DYNAMIC_SET_MAC) &&
71         !iface_tool_.lock()->SetUpState(iface_name.c_str(), true)) {
72         LOG(ERROR) << "SetUpState(true) failed. Wait for driver ready.";
73         // Wait for driver ready and try to set iface UP again
74         if (legacy_hal_.lock()->waitForDriverReady() != legacy_hal::WIFI_SUCCESS) {
75             LOG(ERROR) << "SetUpState(true) wait for driver ready failed.";
76             return false;
77         }
78         if (!iface_tool_.lock()->SetUpState(iface_name.c_str(), true)) {
79             LOG(ERROR) << "SetUpState(true) failed after retry.";
80             return false;
81         }
82     }
83 #endif
84     IfaceEventHandlers event_handlers = {};
85     const auto it = event_handlers_map_.find(iface_name);
86     if (it != event_handlers_map_.end()) {
87         event_handlers = it->second;
88     }
89     if (event_handlers.on_state_toggle_off_on != nullptr) {
90         event_handlers.on_state_toggle_off_on(iface_name);
91     }
92     if (!success) {
93         LOG(ERROR) << "SetMacAddress failed on " << iface_name;
94     } else {
95         LOG(DEBUG) << "SetMacAddress succeeded on " << iface_name;
96     }
97     return success;
98 }
99 
getOrCreateRandomMacAddress()100 std::array<uint8_t, 6> WifiIfaceUtil::getOrCreateRandomMacAddress() {
101     if (random_mac_address_) {
102         return *random_mac_address_.get();
103     }
104     random_mac_address_ = std::make_unique<std::array<uint8_t, 6>>(createRandomMacAddress());
105     return *random_mac_address_.get();
106 }
107 
registerIfaceEventHandlers(const std::string & iface_name,IfaceEventHandlers handlers)108 void WifiIfaceUtil::registerIfaceEventHandlers(const std::string& iface_name,
109                                                IfaceEventHandlers handlers) {
110     event_handlers_map_[iface_name] = handlers;
111 }
112 
unregisterIfaceEventHandlers(const std::string & iface_name)113 void WifiIfaceUtil::unregisterIfaceEventHandlers(const std::string& iface_name) {
114     event_handlers_map_.erase(iface_name);
115 }
116 
createRandomMacAddress()117 std::array<uint8_t, 6> WifiIfaceUtil::createRandomMacAddress() {
118     std::array<uint8_t, 6> address = {};
119     std::random_device rd;
120     std::default_random_engine engine(rd());
121     std::uniform_int_distribution<uint8_t> dist(std::numeric_limits<uint8_t>::min(),
122                                                 std::numeric_limits<uint8_t>::max());
123     for (size_t i = 0; i < address.size(); i++) {
124         address[i] = dist(engine);
125     }
126     // Set the local bit and clear the multicast bit.
127     address[0] |= kMacAddressLocallyAssignedMask;
128     address[0] &= ~kMacAddressMulticastMask;
129     return address;
130 }
131 
setUpState(const std::string & iface_name,bool request_up)132 bool WifiIfaceUtil::setUpState(const std::string& iface_name, bool request_up) {
133     if (!iface_tool_.lock()->SetUpState(iface_name.c_str(), request_up)) {
134         LOG(ERROR) << "SetUpState to " << request_up << " failed";
135         return false;
136     }
137     return true;
138 }
139 
ifNameToIndex(const std::string & iface_name)140 unsigned WifiIfaceUtil::ifNameToIndex(const std::string& iface_name) {
141     return if_nametoindex(iface_name.c_str());
142 }
143 
createBridge(const std::string & br_name)144 bool WifiIfaceUtil::createBridge(const std::string& br_name) {
145     if (!iface_tool_.lock()->createBridge(br_name)) {
146         return false;
147     }
148 
149     if (!iface_tool_.lock()->SetUpState(br_name.c_str(), true)) {
150         LOG(ERROR) << "bridge SetUpState(true) failed.";
151     }
152     return true;
153 }
154 
deleteBridge(const std::string & br_name)155 bool WifiIfaceUtil::deleteBridge(const std::string& br_name) {
156     if (!iface_tool_.lock()->SetUpState(br_name.c_str(), false)) {
157         LOG(INFO) << "SetUpState(false) failed for bridge=" << br_name.c_str();
158     }
159 
160     return iface_tool_.lock()->deleteBridge(br_name);
161 }
162 
addIfaceToBridge(const std::string & br_name,const std::string & if_name)163 bool WifiIfaceUtil::addIfaceToBridge(const std::string& br_name, const std::string& if_name) {
164     return iface_tool_.lock()->addIfaceToBridge(br_name, if_name);
165 }
166 
removeIfaceFromBridge(const std::string & br_name,const std::string & if_name)167 bool WifiIfaceUtil::removeIfaceFromBridge(const std::string& br_name, const std::string& if_name) {
168     return iface_tool_.lock()->removeIfaceFromBridge(br_name, if_name);
169 }
170 
171 }  // namespace iface_util
172 }  // namespace implementation
173 }  // namespace V1_6
174 }  // namespace wifi
175 }  // namespace hardware
176 }  // namespace android
177