blob: 28b7f9a21bfaba1c15d8e55bb63c892131cf12f1 [file] [log] [blame]
Roshan Piusaabe5752016-09-29 09:03:59 -07001/*
2 * Copyright (C) 2016 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 <array>
18
Roshan Piusaabe5752016-09-29 09:03:59 -070019#include <android-base/logging.h>
20#include <cutils/properties.h>
Roshan Pius908a69a2016-10-03 13:33:23 -070021#include <wifi_system/interface_tool.h>
Roshan Piusaabe5752016-09-29 09:03:59 -070022
Roshan Pius955542e2016-10-28 09:42:44 -070023#include "wifi_legacy_hal.h"
24
25namespace android {
26namespace hardware {
27namespace wifi {
28namespace V1_0 {
29namespace implementation {
30namespace legacy_hal {
31// Constants used in the class.
Roshan Pius511cc492016-10-28 09:54:26 -070032static constexpr uint32_t kMaxVersionStringLength = 256;
33
Roshan Piusaabe5752016-09-29 09:03:59 -070034// Legacy HAL functions accept "C" style function pointers, so use global
35// functions to pass to the legacy HAL function and store the corresponding
36// std::function methods to be invoked.
37// Callback to be invoked once |stop| is complete.
38std::function<void(wifi_handle handle)> on_stop_complete_internal_callback;
39void onStopComplete(wifi_handle handle) {
40 if (on_stop_complete_internal_callback) {
41 on_stop_complete_internal_callback(handle);
42 }
43}
Roshan Piuscdb77f32016-10-03 14:09:57 -070044
45// Callback to be invoked for driver dump.
46std::function<void(char*, int)> on_driver_memory_dump_internal_callback;
47void onDriverMemoryDump(char* buffer, int buffer_size) {
48 if (on_driver_memory_dump_internal_callback) {
49 on_driver_memory_dump_internal_callback(buffer, buffer_size);
50 }
51}
52
53// Callback to be invoked for firmware dump.
54std::function<void(char*, int)> on_firmware_memory_dump_internal_callback;
55void onFirmwareMemoryDump(char* buffer, int buffer_size) {
56 if (on_firmware_memory_dump_internal_callback) {
57 on_firmware_memory_dump_internal_callback(buffer, buffer_size);
58 }
59}
Roshan Pius955542e2016-10-28 09:42:44 -070060// End of the free-standing "C" style callbacks.
Roshan Piusaabe5752016-09-29 09:03:59 -070061
Roshan Piusaabe5752016-09-29 09:03:59 -070062WifiLegacyHal::WifiLegacyHal()
63 : global_handle_(nullptr),
64 wlan_interface_handle_(nullptr),
Roshan Pius908a69a2016-10-03 13:33:23 -070065 awaiting_event_loop_termination_(false) {}
Roshan Piusaabe5752016-09-29 09:03:59 -070066
67wifi_error WifiLegacyHal::start() {
68 // Ensure that we're starting in a good state.
69 CHECK(!global_handle_ && !wlan_interface_handle_ &&
70 !awaiting_event_loop_termination_);
71
Roshan Pius908a69a2016-10-03 13:33:23 -070072 android::wifi_system::InterfaceTool if_tool;
Roshan Pius955542e2016-10-28 09:42:44 -070073 // TODO: Add back the HAL Tool if we need to. All we need from the HAL tool
74 // for now is this function call which we can directly call.
75 wifi_error status = init_wifi_vendor_hal_func_table(&global_func_table_);
76 if (status != WIFI_SUCCESS) {
Roshan Pius908a69a2016-10-03 13:33:23 -070077 LOG(ERROR) << "Failed to initialize legacy hal function table";
78 return WIFI_ERROR_UNKNOWN;
79 }
80 if (!if_tool.SetWifiUpState(true)) {
81 LOG(ERROR) << "Failed to set WiFi interface up";
82 return WIFI_ERROR_UNKNOWN;
83 }
84
Roshan Piusaabe5752016-09-29 09:03:59 -070085 LOG(INFO) << "Starting legacy HAL";
Roshan Pius955542e2016-10-28 09:42:44 -070086 status = global_func_table_.wifi_initialize(&global_handle_);
Roshan Piusaabe5752016-09-29 09:03:59 -070087 if (status != WIFI_SUCCESS || !global_handle_) {
88 LOG(ERROR) << "Failed to retrieve global handle";
89 return status;
90 }
91 event_loop_thread_ = std::thread(&WifiLegacyHal::runEventLoop, this);
92 status = retrieveWlanInterfaceHandle();
93 if (status != WIFI_SUCCESS || !wlan_interface_handle_) {
94 LOG(ERROR) << "Failed to retrieve wlan interface handle";
95 return status;
96 }
97 LOG(VERBOSE) << "Legacy HAL start complete";
98 return WIFI_SUCCESS;
99}
100
101wifi_error WifiLegacyHal::stop(
102 const std::function<void()>& on_stop_complete_user_callback) {
103 LOG(INFO) << "Stopping legacy HAL";
104 on_stop_complete_internal_callback = [&](wifi_handle handle) {
105 CHECK_EQ(global_handle_, handle) << "Handle mismatch";
106 on_stop_complete_user_callback();
Roshan Pius511cc492016-10-28 09:54:26 -0700107 // Invalidate all the internal pointers now that the HAL is
108 // stopped.
109 invalidate();
Roshan Piusaabe5752016-09-29 09:03:59 -0700110 };
111 awaiting_event_loop_termination_ = true;
112 global_func_table_.wifi_cleanup(global_handle_, onStopComplete);
113 LOG(VERBOSE) << "Legacy HAL stop initiated";
114 return WIFI_SUCCESS;
115}
116
Roshan Piusab5c4712016-10-06 14:37:15 -0700117std::string WifiLegacyHal::getApIfaceName() {
118 // Fake name. This interface does not exist in legacy HAL
119 // API's.
120 return "ap0";
121}
122
123std::string WifiLegacyHal::getNanIfaceName() {
124 // Fake name. This interface does not exist in legacy HAL
125 // API's.
126 return "nan0";
127}
128
129std::string WifiLegacyHal::getP2pIfaceName() {
130 std::array<char, PROPERTY_VALUE_MAX> buffer;
131 property_get("wifi.direct.interface", buffer.data(), "p2p0");
132 return buffer.data();
133}
134
135std::string WifiLegacyHal::getStaIfaceName() {
136 std::array<char, PROPERTY_VALUE_MAX> buffer;
137 property_get("wifi.interface", buffer.data(), "wlan0");
138 return buffer.data();
139}
140
141std::pair<wifi_error, std::string> WifiLegacyHal::getDriverVersion() {
Roshan Pius4b26c832016-10-03 12:49:58 -0700142 std::array<char, kMaxVersionStringLength> buffer;
143 buffer.fill(0);
144 wifi_error status = global_func_table_.wifi_get_driver_version(
145 wlan_interface_handle_, buffer.data(), buffer.size());
Roshan Pius0a47c182016-10-28 10:23:00 -0700146 return {status, buffer.data()};
Roshan Pius4b26c832016-10-03 12:49:58 -0700147}
148
Roshan Piusab5c4712016-10-06 14:37:15 -0700149std::pair<wifi_error, std::string> WifiLegacyHal::getFirmwareVersion() {
Roshan Pius4b26c832016-10-03 12:49:58 -0700150 std::array<char, kMaxVersionStringLength> buffer;
151 buffer.fill(0);
152 wifi_error status = global_func_table_.wifi_get_firmware_version(
153 wlan_interface_handle_, buffer.data(), buffer.size());
Roshan Pius0a47c182016-10-28 10:23:00 -0700154 return {status, buffer.data()};
Roshan Pius4b26c832016-10-03 12:49:58 -0700155}
156
Roshan Pius3c868522016-10-27 12:43:49 -0700157std::pair<wifi_error, std::vector<uint8_t>>
Roshan Piusab5c4712016-10-06 14:37:15 -0700158WifiLegacyHal::requestDriverMemoryDump() {
Roshan Pius3c868522016-10-27 12:43:49 -0700159 std::vector<uint8_t> driver_dump;
Roshan Piuscdb77f32016-10-03 14:09:57 -0700160 on_driver_memory_dump_internal_callback = [&driver_dump](char* buffer,
161 int buffer_size) {
Roshan Pius3c868522016-10-27 12:43:49 -0700162 driver_dump.insert(driver_dump.end(),
163 reinterpret_cast<uint8_t*>(buffer),
164 reinterpret_cast<uint8_t*>(buffer) + buffer_size);
Roshan Piuscdb77f32016-10-03 14:09:57 -0700165 };
166 wifi_error status = global_func_table_.wifi_get_driver_memory_dump(
167 wlan_interface_handle_, {onDriverMemoryDump});
168 on_driver_memory_dump_internal_callback = nullptr;
Roshan Pius0a47c182016-10-28 10:23:00 -0700169 return {status, std::move(driver_dump)};
Roshan Piuscdb77f32016-10-03 14:09:57 -0700170}
171
Roshan Pius3c868522016-10-27 12:43:49 -0700172std::pair<wifi_error, std::vector<uint8_t>>
Roshan Piusab5c4712016-10-06 14:37:15 -0700173WifiLegacyHal::requestFirmwareMemoryDump() {
Roshan Pius3c868522016-10-27 12:43:49 -0700174 std::vector<uint8_t> firmware_dump;
Roshan Piuscdb77f32016-10-03 14:09:57 -0700175 on_firmware_memory_dump_internal_callback = [&firmware_dump](
176 char* buffer, int buffer_size) {
Roshan Pius3c868522016-10-27 12:43:49 -0700177 firmware_dump.insert(firmware_dump.end(),
178 reinterpret_cast<uint8_t*>(buffer),
179 reinterpret_cast<uint8_t*>(buffer) + buffer_size);
Roshan Piuscdb77f32016-10-03 14:09:57 -0700180 };
181 wifi_error status = global_func_table_.wifi_get_firmware_memory_dump(
182 wlan_interface_handle_, {onFirmwareMemoryDump});
183 on_firmware_memory_dump_internal_callback = nullptr;
Roshan Pius0a47c182016-10-28 10:23:00 -0700184 return {status, std::move(firmware_dump)};
185}
186
187std::pair<wifi_error, uint32_t> WifiLegacyHal::getSupportedFeatureSet() {
188 feature_set set;
189 static_assert(sizeof(set) == sizeof(uint32_t),
190 "Some features can not be represented in output");
191 wifi_error status = global_func_table_.wifi_get_supported_feature_set(
192 wlan_interface_handle_, &set);
193 return {status, static_cast<uint32_t>(set)};
194}
195
196std::pair<wifi_error, PacketFilterCapabilities>
197WifiLegacyHal::getPacketFilterCapabilities() {
198 PacketFilterCapabilities caps;
199 wifi_error status = global_func_table_.wifi_get_packet_filter_capabilities(
200 wlan_interface_handle_, &caps.version, &caps.max_len);
201 return {status, caps};
202}
203
204wifi_error WifiLegacyHal::setPacketFilter(const std::vector<uint8_t>& program) {
205 return global_func_table_.wifi_set_packet_filter(
206 wlan_interface_handle_, program.data(), program.size());
Roshan Piuscdb77f32016-10-03 14:09:57 -0700207}
208
Roshan Piusaabe5752016-09-29 09:03:59 -0700209wifi_error WifiLegacyHal::retrieveWlanInterfaceHandle() {
Roshan Piusab5c4712016-10-06 14:37:15 -0700210 const std::string& ifname_to_find = getStaIfaceName();
Roshan Piusaabe5752016-09-29 09:03:59 -0700211 wifi_interface_handle* iface_handles = nullptr;
212 int num_iface_handles = 0;
213 wifi_error status = global_func_table_.wifi_get_ifaces(
214 global_handle_, &num_iface_handles, &iface_handles);
215 if (status != WIFI_SUCCESS) {
Roshan Pius511cc492016-10-28 09:54:26 -0700216 LOG(ERROR) << "Failed to enumerate interface handles";
Roshan Piusaabe5752016-09-29 09:03:59 -0700217 return status;
218 }
219 for (int i = 0; i < num_iface_handles; ++i) {
220 std::array<char, IFNAMSIZ> current_ifname;
221 current_ifname.fill(0);
222 status = global_func_table_.wifi_get_iface_name(
223 iface_handles[i], current_ifname.data(), current_ifname.size());
224 if (status != WIFI_SUCCESS) {
Roshan Pius511cc492016-10-28 09:54:26 -0700225 LOG(WARNING) << "Failed to get interface handle name";
Roshan Piusaabe5752016-09-29 09:03:59 -0700226 continue;
227 }
228 if (ifname_to_find == current_ifname.data()) {
229 wlan_interface_handle_ = iface_handles[i];
230 return WIFI_SUCCESS;
231 }
232 }
233 return WIFI_ERROR_UNKNOWN;
234}
235
236void WifiLegacyHal::runEventLoop() {
237 LOG(VERBOSE) << "Starting legacy HAL event loop";
238 global_func_table_.wifi_event_loop(global_handle_);
239 if (!awaiting_event_loop_termination_) {
240 LOG(FATAL) << "Legacy HAL event loop terminated, but HAL was not stopping";
241 }
242 LOG(VERBOSE) << "Legacy HAL event loop terminated";
243 awaiting_event_loop_termination_ = false;
Roshan Pius908a69a2016-10-03 13:33:23 -0700244 android::wifi_system::InterfaceTool if_tool;
245 if_tool.SetWifiUpState(false);
Roshan Piusaabe5752016-09-29 09:03:59 -0700246}
247
Roshan Pius511cc492016-10-28 09:54:26 -0700248void WifiLegacyHal::invalidate() {
249 global_handle_ = nullptr;
250 wlan_interface_handle_ = nullptr;
251 on_stop_complete_internal_callback = nullptr;
252 on_driver_memory_dump_internal_callback = nullptr;
253 on_firmware_memory_dump_internal_callback = nullptr;
254}
Roshan Pius955542e2016-10-28 09:42:44 -0700255
256} // namespace legacy_hal
Roshan Piusaabe5752016-09-29 09:03:59 -0700257} // namespace implementation
258} // namespace V1_0
259} // namespace wifi
260} // namespace hardware
261} // namespace android