2022-07-15 23:25:39 +00:00
|
|
|
/*
|
|
|
|
|
* Copyright (C) 2022 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 "wifi.h"
|
|
|
|
|
|
2023-04-13 12:14:11 -04:00
|
|
|
#include <android-base/file.h>
|
2022-07-15 23:25:39 +00:00
|
|
|
#include <android-base/logging.h>
|
2023-10-21 03:16:10 +08:00
|
|
|
#include <cutils/properties.h>
|
|
|
|
|
#include <sys/stat.h>
|
|
|
|
|
#include <sys/sysmacros.h>
|
2022-07-15 23:25:39 +00:00
|
|
|
|
|
|
|
|
#include "aidl_return_util.h"
|
2023-04-10 23:47:45 +00:00
|
|
|
#include "aidl_sync_util.h"
|
2022-07-15 23:25:39 +00:00
|
|
|
#include "wifi_status_util.h"
|
|
|
|
|
|
|
|
|
|
namespace {
|
2023-10-21 03:16:10 +08:00
|
|
|
using android::base::unique_fd;
|
|
|
|
|
|
2022-07-15 23:25:39 +00:00
|
|
|
// Starting Chip ID, will be assigned to primary chip
|
|
|
|
|
static constexpr int32_t kPrimaryChipId = 0;
|
2023-10-21 03:16:10 +08:00
|
|
|
constexpr char kCpioMagic[] = "070701";
|
|
|
|
|
constexpr char kTombstoneFolderPath[] = "/data/vendor/tombstones/wifi/";
|
|
|
|
|
|
|
|
|
|
// Helper function for |cpioArchiveFilesInDir|
|
|
|
|
|
bool cpioWriteHeader(int out_fd, struct stat& st, const char* file_name, size_t file_name_len) {
|
|
|
|
|
const int buf_size = 32 * 1024;
|
|
|
|
|
std::array<char, buf_size> read_buf;
|
|
|
|
|
ssize_t llen = snprintf(
|
|
|
|
|
read_buf.data(), buf_size, "%s%08X%08X%08X%08X%08X%08X%08X%08X%08X%08X%08X%08X%08X",
|
|
|
|
|
kCpioMagic, static_cast<int>(st.st_ino), st.st_mode, st.st_uid, st.st_gid,
|
|
|
|
|
static_cast<int>(st.st_nlink), static_cast<int>(st.st_mtime),
|
|
|
|
|
static_cast<int>(st.st_size), major(st.st_dev), minor(st.st_dev), major(st.st_rdev),
|
|
|
|
|
minor(st.st_rdev), static_cast<uint32_t>(file_name_len), 0);
|
|
|
|
|
if (write(out_fd, read_buf.data(), llen < buf_size ? llen : buf_size - 1) == -1) {
|
|
|
|
|
PLOG(ERROR) << "Error writing cpio header to file " << file_name;
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
if (write(out_fd, file_name, file_name_len) == -1) {
|
|
|
|
|
PLOG(ERROR) << "Error writing filename to file " << file_name;
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// NUL Pad header up to 4 multiple bytes.
|
|
|
|
|
llen = (llen + file_name_len) % 4;
|
|
|
|
|
if (llen != 0) {
|
|
|
|
|
const uint32_t zero = 0;
|
|
|
|
|
if (write(out_fd, &zero, 4 - llen) == -1) {
|
|
|
|
|
PLOG(ERROR) << "Error padding 0s to file " << file_name;
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Helper function for |cpioArchiveFilesInDir|
|
|
|
|
|
size_t cpioWriteFileContent(int fd_read, int out_fd, struct stat& st) {
|
|
|
|
|
// writing content of file
|
|
|
|
|
std::array<char, 32 * 1024> read_buf;
|
|
|
|
|
ssize_t llen = st.st_size;
|
|
|
|
|
size_t n_error = 0;
|
|
|
|
|
while (llen > 0) {
|
|
|
|
|
ssize_t bytes_read = read(fd_read, read_buf.data(), read_buf.size());
|
|
|
|
|
if (bytes_read == -1) {
|
|
|
|
|
PLOG(ERROR) << "Error reading file";
|
|
|
|
|
return ++n_error;
|
|
|
|
|
}
|
|
|
|
|
llen -= bytes_read;
|
|
|
|
|
if (write(out_fd, read_buf.data(), bytes_read) == -1) {
|
|
|
|
|
PLOG(ERROR) << "Error writing data to file";
|
|
|
|
|
return ++n_error;
|
|
|
|
|
}
|
|
|
|
|
if (bytes_read == 0) { // this should never happen, but just in case
|
|
|
|
|
// to unstuck from while loop
|
|
|
|
|
PLOG(ERROR) << "Unexpected read result";
|
|
|
|
|
n_error++;
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
llen = st.st_size % 4;
|
|
|
|
|
if (llen != 0) {
|
|
|
|
|
const uint32_t zero = 0;
|
|
|
|
|
if (write(out_fd, &zero, 4 - llen) == -1) {
|
|
|
|
|
PLOG(ERROR) << "Error padding 0s to file";
|
|
|
|
|
return ++n_error;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
return n_error;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Helper function for |cpioArchiveFilesInDir|
|
|
|
|
|
bool cpioWriteFileTrailer(int out_fd) {
|
|
|
|
|
const int buf_size = 4096;
|
|
|
|
|
std::array<char, buf_size> read_buf;
|
|
|
|
|
read_buf.fill(0);
|
|
|
|
|
ssize_t llen = snprintf(read_buf.data(), 4096, "070701%040X%056X%08XTRAILER!!!", 1, 0x0b, 0);
|
|
|
|
|
if (write(out_fd, read_buf.data(), (llen < buf_size ? llen : buf_size - 1) + 4) == -1) {
|
|
|
|
|
PLOG(ERROR) << "Error writing trailing bytes";
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Archives all files in |input_dir| and writes result into |out_fd|
|
|
|
|
|
// Logic obtained from //external/toybox/toys/posix/cpio.c "Output cpio archive"
|
|
|
|
|
// portion
|
|
|
|
|
size_t cpioArchiveFilesInDir(int out_fd, const char* input_dir) {
|
|
|
|
|
struct dirent* dp;
|
|
|
|
|
size_t n_error = 0;
|
|
|
|
|
std::unique_ptr<DIR, decltype(&closedir)> dir_dump(opendir(input_dir), closedir);
|
|
|
|
|
if (!dir_dump) {
|
|
|
|
|
PLOG(ERROR) << "Failed to open directory";
|
|
|
|
|
return ++n_error;
|
|
|
|
|
}
|
|
|
|
|
while ((dp = readdir(dir_dump.get()))) {
|
|
|
|
|
if (dp->d_type != DT_REG) {
|
|
|
|
|
continue;
|
|
|
|
|
}
|
|
|
|
|
std::string cur_file_name(dp->d_name);
|
|
|
|
|
struct stat st;
|
|
|
|
|
const std::string cur_file_path = kTombstoneFolderPath + cur_file_name;
|
|
|
|
|
if (stat(cur_file_path.c_str(), &st) == -1) {
|
|
|
|
|
PLOG(ERROR) << "Failed to get file stat for " << cur_file_path;
|
|
|
|
|
n_error++;
|
|
|
|
|
continue;
|
|
|
|
|
}
|
|
|
|
|
const int fd_read = open(cur_file_path.c_str(), O_RDONLY);
|
|
|
|
|
if (fd_read == -1) {
|
|
|
|
|
PLOG(ERROR) << "Failed to open file " << cur_file_path;
|
|
|
|
|
n_error++;
|
|
|
|
|
continue;
|
|
|
|
|
}
|
|
|
|
|
std::string file_name_with_last_modified_time =
|
|
|
|
|
cur_file_name + "-" + std::to_string(st.st_mtime);
|
|
|
|
|
// string.size() does not include the null terminator. The cpio FreeBSD
|
|
|
|
|
// file header expects the null character to be included in the length.
|
|
|
|
|
const size_t file_name_len = file_name_with_last_modified_time.size() + 1;
|
|
|
|
|
unique_fd file_auto_closer(fd_read);
|
|
|
|
|
if (!cpioWriteHeader(out_fd, st, file_name_with_last_modified_time.c_str(),
|
|
|
|
|
file_name_len)) {
|
|
|
|
|
return ++n_error;
|
|
|
|
|
}
|
|
|
|
|
size_t write_error = cpioWriteFileContent(fd_read, out_fd, st);
|
|
|
|
|
if (write_error) {
|
|
|
|
|
return n_error + write_error;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
if (!cpioWriteFileTrailer(out_fd)) {
|
|
|
|
|
return ++n_error;
|
|
|
|
|
}
|
|
|
|
|
return n_error;
|
|
|
|
|
}
|
|
|
|
|
|
2022-07-15 23:25:39 +00:00
|
|
|
} // namespace
|
|
|
|
|
|
|
|
|
|
namespace aidl {
|
|
|
|
|
namespace android {
|
|
|
|
|
namespace hardware {
|
|
|
|
|
namespace wifi {
|
|
|
|
|
using aidl_return_util::validateAndCall;
|
|
|
|
|
using aidl_return_util::validateAndCallWithLock;
|
2023-04-10 23:47:45 +00:00
|
|
|
using aidl_sync_util::acquireGlobalLock;
|
2022-07-15 23:25:39 +00:00
|
|
|
|
|
|
|
|
Wifi::Wifi(const std::shared_ptr<::android::wifi_system::InterfaceTool> iface_tool,
|
|
|
|
|
const std::shared_ptr<legacy_hal::WifiLegacyHalFactory> legacy_hal_factory,
|
|
|
|
|
const std::shared_ptr<mode_controller::WifiModeController> mode_controller,
|
|
|
|
|
const std::shared_ptr<feature_flags::WifiFeatureFlags> feature_flags)
|
|
|
|
|
: iface_tool_(iface_tool),
|
|
|
|
|
legacy_hal_factory_(legacy_hal_factory),
|
|
|
|
|
mode_controller_(mode_controller),
|
|
|
|
|
feature_flags_(feature_flags),
|
|
|
|
|
run_state_(RunState::STOPPED) {}
|
|
|
|
|
|
|
|
|
|
bool Wifi::isValid() {
|
|
|
|
|
// This object is always valid.
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
ndk::ScopedAStatus Wifi::registerEventCallback(
|
|
|
|
|
const std::shared_ptr<IWifiEventCallback>& in_callback) {
|
|
|
|
|
return validateAndCall(this, WifiStatusCode::ERROR_UNKNOWN,
|
|
|
|
|
&Wifi::registerEventCallbackInternal, in_callback);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
ndk::ScopedAStatus Wifi::isStarted(bool* _aidl_return) {
|
|
|
|
|
*_aidl_return = (run_state_ != RunState::STOPPED);
|
|
|
|
|
return ndk::ScopedAStatus::ok();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
ndk::ScopedAStatus Wifi::start() {
|
|
|
|
|
return validateAndCall(this, WifiStatusCode::ERROR_UNKNOWN, &Wifi::startInternal);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
ndk::ScopedAStatus Wifi::stop() {
|
|
|
|
|
return validateAndCallWithLock(this, WifiStatusCode::ERROR_UNKNOWN, &Wifi::stopInternal);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
ndk::ScopedAStatus Wifi::getChipIds(std::vector<int32_t>* _aidl_return) {
|
|
|
|
|
return validateAndCall(this, WifiStatusCode::ERROR_UNKNOWN, &Wifi::getChipIdsInternal,
|
|
|
|
|
_aidl_return);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
ndk::ScopedAStatus Wifi::getChip(int32_t in_chipId, std::shared_ptr<IWifiChip>* _aidl_return) {
|
|
|
|
|
return validateAndCall(this, WifiStatusCode::ERROR_UNKNOWN, &Wifi::getChipInternal,
|
|
|
|
|
_aidl_return, in_chipId);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
binder_status_t Wifi::dump(int fd, const char** args, uint32_t numArgs) {
|
2023-04-10 23:47:45 +00:00
|
|
|
const auto lock = acquireGlobalLock();
|
2022-07-15 23:25:39 +00:00
|
|
|
LOG(INFO) << "-----------Debug was called----------------";
|
2023-10-21 03:16:10 +08:00
|
|
|
if (chips_.size() != 0) {
|
|
|
|
|
for (std::shared_ptr<WifiChip> chip : chips_) {
|
|
|
|
|
if (!chip.get()) continue;
|
|
|
|
|
chip->dump(fd, args, numArgs);
|
|
|
|
|
}
|
2022-07-15 23:25:39 +00:00
|
|
|
}
|
2023-10-21 03:16:10 +08:00
|
|
|
uint32_t n_error = cpioArchiveFilesInDir(fd, kTombstoneFolderPath);
|
|
|
|
|
if (n_error != 0) {
|
|
|
|
|
LOG(ERROR) << n_error << " errors occurred in cpio function";
|
2022-07-15 23:25:39 +00:00
|
|
|
}
|
2023-04-13 12:14:11 -04:00
|
|
|
::android::base::WriteStringToFd("\n", fd);
|
2023-10-21 03:16:10 +08:00
|
|
|
fsync(fd);
|
2022-07-15 23:25:39 +00:00
|
|
|
return STATUS_OK;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
ndk::ScopedAStatus Wifi::registerEventCallbackInternal(
|
|
|
|
|
const std::shared_ptr<IWifiEventCallback>& event_callback) {
|
|
|
|
|
if (!event_cb_handler_.addCallback(event_callback)) {
|
|
|
|
|
return createWifiStatus(WifiStatusCode::ERROR_UNKNOWN);
|
|
|
|
|
}
|
|
|
|
|
return ndk::ScopedAStatus::ok();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
ndk::ScopedAStatus Wifi::startInternal() {
|
|
|
|
|
if (run_state_ == RunState::STARTED) {
|
|
|
|
|
return ndk::ScopedAStatus::ok();
|
|
|
|
|
} else if (run_state_ == RunState::STOPPING) {
|
|
|
|
|
return createWifiStatus(WifiStatusCode::ERROR_NOT_AVAILABLE, "HAL is stopping");
|
|
|
|
|
}
|
|
|
|
|
ndk::ScopedAStatus wifi_status = initializeModeControllerAndLegacyHal();
|
|
|
|
|
if (wifi_status.isOk()) {
|
|
|
|
|
// Register the callback for subsystem restart
|
|
|
|
|
const auto& on_subsystem_restart_callback = [this](const std::string& error) {
|
|
|
|
|
ndk::ScopedAStatus wifi_status = createWifiStatus(WifiStatusCode::ERROR_UNKNOWN, error);
|
|
|
|
|
for (const auto& callback : event_cb_handler_.getCallbacks()) {
|
|
|
|
|
LOG(INFO) << "Attempting to invoke onSubsystemRestart "
|
|
|
|
|
"callback";
|
|
|
|
|
WifiStatusCode errorCode =
|
|
|
|
|
static_cast<WifiStatusCode>(wifi_status.getServiceSpecificError());
|
|
|
|
|
if (!callback->onSubsystemRestart(errorCode).isOk()) {
|
|
|
|
|
LOG(ERROR) << "Failed to invoke onSubsystemRestart callback";
|
|
|
|
|
} else {
|
|
|
|
|
LOG(INFO) << "Succeeded to invoke onSubsystemRestart "
|
|
|
|
|
"callback";
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
// Create the chip instance once the HAL is started.
|
|
|
|
|
int32_t chipId = kPrimaryChipId;
|
|
|
|
|
for (auto& hal : legacy_hals_) {
|
|
|
|
|
chips_.push_back(
|
|
|
|
|
WifiChip::create(chipId, chipId == kPrimaryChipId, hal, mode_controller_,
|
|
|
|
|
std::make_shared<iface_util::WifiIfaceUtil>(iface_tool_, hal),
|
2023-06-14 22:42:07 +00:00
|
|
|
feature_flags_, on_subsystem_restart_callback, false));
|
2022-07-15 23:25:39 +00:00
|
|
|
chipId++;
|
|
|
|
|
}
|
|
|
|
|
run_state_ = RunState::STARTED;
|
|
|
|
|
for (const auto& callback : event_cb_handler_.getCallbacks()) {
|
|
|
|
|
if (!callback->onStart().isOk()) {
|
|
|
|
|
LOG(ERROR) << "Failed to invoke onStart callback";
|
|
|
|
|
};
|
|
|
|
|
}
|
|
|
|
|
LOG(INFO) << "Wifi HAL started";
|
|
|
|
|
} else {
|
|
|
|
|
for (const auto& callback : event_cb_handler_.getCallbacks()) {
|
|
|
|
|
WifiStatusCode errorCode =
|
|
|
|
|
static_cast<WifiStatusCode>(wifi_status.getServiceSpecificError());
|
|
|
|
|
if (!callback->onFailure(errorCode).isOk()) {
|
|
|
|
|
LOG(ERROR) << "Failed to invoke onFailure callback";
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
LOG(ERROR) << "Wifi HAL start failed";
|
|
|
|
|
// Clear the event callback objects since the HAL start failed.
|
|
|
|
|
event_cb_handler_.invalidate();
|
|
|
|
|
}
|
|
|
|
|
return wifi_status;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
ndk::ScopedAStatus Wifi::stopInternal(
|
|
|
|
|
/* NONNULL */ std::unique_lock<std::recursive_mutex>* lock) {
|
|
|
|
|
if (run_state_ == RunState::STOPPED) {
|
|
|
|
|
return ndk::ScopedAStatus::ok();
|
|
|
|
|
} else if (run_state_ == RunState::STOPPING) {
|
|
|
|
|
return createWifiStatus(WifiStatusCode::ERROR_NOT_AVAILABLE, "HAL is stopping");
|
|
|
|
|
}
|
|
|
|
|
// Clear the chip object and its child objects since the HAL is now
|
|
|
|
|
// stopped.
|
|
|
|
|
for (auto& chip : chips_) {
|
|
|
|
|
if (chip.get()) {
|
|
|
|
|
chip->invalidate();
|
|
|
|
|
chip.reset();
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
chips_.clear();
|
|
|
|
|
ndk::ScopedAStatus wifi_status = stopLegacyHalAndDeinitializeModeController(lock);
|
|
|
|
|
if (wifi_status.isOk()) {
|
|
|
|
|
for (const auto& callback : event_cb_handler_.getCallbacks()) {
|
|
|
|
|
if (!callback->onStop().isOk()) {
|
|
|
|
|
LOG(ERROR) << "Failed to invoke onStop callback";
|
|
|
|
|
};
|
|
|
|
|
}
|
|
|
|
|
LOG(INFO) << "Wifi HAL stopped";
|
|
|
|
|
} else {
|
|
|
|
|
for (const auto& callback : event_cb_handler_.getCallbacks()) {
|
|
|
|
|
WifiStatusCode errorCode =
|
|
|
|
|
static_cast<WifiStatusCode>(wifi_status.getServiceSpecificError());
|
|
|
|
|
if (!callback->onFailure(errorCode).isOk()) {
|
|
|
|
|
LOG(ERROR) << "Failed to invoke onFailure callback";
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
LOG(ERROR) << "Wifi HAL stop failed";
|
|
|
|
|
}
|
|
|
|
|
// Clear the event callback objects since the HAL is now stopped.
|
|
|
|
|
event_cb_handler_.invalidate();
|
|
|
|
|
return wifi_status;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
std::pair<std::vector<int32_t>, ndk::ScopedAStatus> Wifi::getChipIdsInternal() {
|
|
|
|
|
std::vector<int32_t> chip_ids;
|
|
|
|
|
|
|
|
|
|
for (auto& chip : chips_) {
|
|
|
|
|
int32_t chip_id = getChipIdFromWifiChip(chip);
|
|
|
|
|
if (chip_id != INT32_MAX) chip_ids.emplace_back(chip_id);
|
|
|
|
|
}
|
|
|
|
|
return {std::move(chip_ids), ndk::ScopedAStatus::ok()};
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
std::pair<std::shared_ptr<IWifiChip>, ndk::ScopedAStatus> Wifi::getChipInternal(int32_t chip_id) {
|
|
|
|
|
for (auto& chip : chips_) {
|
|
|
|
|
int32_t cand_id = getChipIdFromWifiChip(chip);
|
|
|
|
|
if ((cand_id != INT32_MAX) && (cand_id == chip_id)) return {chip, ndk::ScopedAStatus::ok()};
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return {nullptr, createWifiStatus(WifiStatusCode::ERROR_INVALID_ARGS)};
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
ndk::ScopedAStatus Wifi::initializeModeControllerAndLegacyHal() {
|
|
|
|
|
if (!mode_controller_->initialize()) {
|
|
|
|
|
LOG(ERROR) << "Failed to initialize firmware mode controller";
|
|
|
|
|
return createWifiStatus(WifiStatusCode::ERROR_UNKNOWN);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
legacy_hals_ = legacy_hal_factory_->getHals();
|
|
|
|
|
if (legacy_hals_.empty()) return createWifiStatus(WifiStatusCode::ERROR_UNKNOWN);
|
|
|
|
|
int index = 0; // for failure log
|
|
|
|
|
for (auto& hal : legacy_hals_) {
|
|
|
|
|
legacy_hal::wifi_error legacy_status = hal->initialize();
|
|
|
|
|
if (legacy_status != legacy_hal::WIFI_SUCCESS) {
|
|
|
|
|
// Currently WifiLegacyHal::initialize does not allocate extra mem,
|
|
|
|
|
// only initializes the function table. If this changes, need to
|
|
|
|
|
// implement WifiLegacyHal::deinitialize and deinitalize the
|
|
|
|
|
// HALs already initialized
|
|
|
|
|
LOG(ERROR) << "Failed to initialize legacy HAL index: " << index
|
|
|
|
|
<< " error: " << legacyErrorToString(legacy_status);
|
|
|
|
|
return createWifiStatusFromLegacyError(legacy_status);
|
|
|
|
|
}
|
|
|
|
|
index++;
|
|
|
|
|
}
|
|
|
|
|
return ndk::ScopedAStatus::ok();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
ndk::ScopedAStatus Wifi::stopLegacyHalAndDeinitializeModeController(
|
|
|
|
|
/* NONNULL */ std::unique_lock<std::recursive_mutex>* lock) {
|
|
|
|
|
legacy_hal::wifi_error legacy_status = legacy_hal::WIFI_SUCCESS;
|
|
|
|
|
int index = 0;
|
|
|
|
|
|
|
|
|
|
run_state_ = RunState::STOPPING;
|
|
|
|
|
for (auto& hal : legacy_hals_) {
|
|
|
|
|
legacy_hal::wifi_error tmp = hal->stop(lock, [&]() {});
|
|
|
|
|
if (tmp != legacy_hal::WIFI_SUCCESS) {
|
|
|
|
|
LOG(ERROR) << "Failed to stop legacy HAL index: " << index
|
|
|
|
|
<< " error: " << legacyErrorToString(legacy_status);
|
|
|
|
|
legacy_status = tmp;
|
|
|
|
|
}
|
|
|
|
|
index++;
|
|
|
|
|
}
|
|
|
|
|
run_state_ = RunState::STOPPED;
|
|
|
|
|
|
|
|
|
|
if (legacy_status != legacy_hal::WIFI_SUCCESS) {
|
|
|
|
|
LOG(ERROR) << "One or more legacy HALs failed to stop";
|
|
|
|
|
return createWifiStatusFromLegacyError(legacy_status);
|
|
|
|
|
}
|
|
|
|
|
if (!mode_controller_->deinitialize()) {
|
|
|
|
|
LOG(ERROR) << "Failed to deinitialize firmware mode controller";
|
|
|
|
|
return createWifiStatus(WifiStatusCode::ERROR_UNKNOWN);
|
|
|
|
|
}
|
|
|
|
|
return ndk::ScopedAStatus::ok();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
int32_t Wifi::getChipIdFromWifiChip(std::shared_ptr<WifiChip>& chip) {
|
|
|
|
|
int32_t chip_id = INT32_MAX;
|
|
|
|
|
if (chip.get()) {
|
|
|
|
|
ndk::ScopedAStatus status = chip->getId(&chip_id);
|
|
|
|
|
if (!status.isOk()) {
|
|
|
|
|
// Reset value if operation failed.
|
|
|
|
|
chip_id = INT32_MAX;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
return chip_id;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
} // namespace wifi
|
|
|
|
|
} // namespace hardware
|
|
|
|
|
} // namespace android
|
|
|
|
|
} // namespace aidl
|