/*
* 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 <functional>
#include <memory>
#include <vector>
#include <linux/netlink.h>
#include <gtest/gtest.h>
#include "android/net/wifi/IWifiScannerImpl.h"
#include "wificond/net/kernel-header-latest/nl80211.h"
#include "wificond/scanning/scan_result.h"
#include "wificond/scanning/scan_utils.h"
#include "wificond/tests/mock_netlink_manager.h"
using std::bind;
using std::placeholders::_1;
using std::placeholders::_2;
using std::unique_ptr;
using std::vector;
using testing::AllOf;
using testing::Invoke;
using testing::NiceMock;
using testing::Not;
using testing::Return;
using testing::_;
using android::net::wifi::IWifiScannerImpl;
using com::android::server::wifi::wificond::NativeScanResult;
namespace android {
namespace wificond {
namespace {
constexpr uint32_t kFakeInterfaceIndex = 12;
constexpr uint32_t kFakeScheduledScanIntervalMs = 20000;
constexpr uint32_t kFakeSequenceNumber = 1984;
constexpr int kFakeErrorCode = EIO;
constexpr int32_t kFake2gRssiThreshold = -80;
constexpr int32_t kFake5gRssiThreshold = -77;
constexpr bool kFakeUseRandomMAC = true;
constexpr bool kFakeRequestLowPower = true;
constexpr bool kFakeRequestSchedScanRelativeRssi = true;
constexpr int kFakeScanType = IWifiScannerImpl::SCAN_TYPE_LOW_SPAN;
// Currently, control messages are only created by the kernel and sent to us.
// Therefore NL80211Packet doesn't have corresponding constructor.
// For test we manually create control messages using this helper function.
NL80211Packet CreateControlMessageError(int error_code) {
vector<uint8_t> data;
data.resize(NLMSG_HDRLEN + NLA_ALIGN(sizeof(int)), 0);
// Initialize length field.
nlmsghdr* nl_header = reinterpret_cast<nlmsghdr*>(data.data());
nl_header->nlmsg_len = data.size();
nl_header->nlmsg_type = NLMSG_ERROR;
nl_header->nlmsg_seq = kFakeSequenceNumber;
nl_header->nlmsg_pid = getpid();
int* error_field = reinterpret_cast<int*>(data.data() + NLMSG_HDRLEN);
*error_field = -error_code;
return NL80211Packet(data);
}
NL80211Packet CreateControlMessageAck() {
return CreateControlMessageError(0);
}
// This is a helper function to mock the behavior of NetlinkManager::
// SendMessageAndGetResponses() when we expect a single packet response.
// |request_message| and |response| are mapped to existing parameters of
// SendMessageAndGetResponses().
// |mock_response| and |mock_return value| are additional parameters used
// for specifying expected results,
bool AppendMessageAndReturn(
NL80211Packet& mock_response,
bool mock_return_value,
const NL80211Packet& request_message,
vector<std::unique_ptr<const NL80211Packet>>* response) {
response->push_back(std::make_unique<NL80211Packet>(mock_response));
return mock_return_value;
}
} // namespace
class ScanUtilsTest : public ::testing::Test {
protected:
virtual void SetUp() {
ON_CALL(netlink_manager_,
SendMessageAndGetResponses(_, _)).WillByDefault(Return(true));
}
NiceMock<MockNetlinkManager> netlink_manager_;
ScanUtils scan_utils_{&netlink_manager_};
};
MATCHER_P(DoesNL80211PacketMatchCommand, command,
"Check if the netlink packet matches |command|") {
return arg.GetCommand() == command;
}
MATCHER_P(DoesNL80211PacketHaveAttribute, attr,
"Check if the netlink packet has atttribute |attr|") {
return arg.HasAttribute(attr);
}
MATCHER_P2(DoesNL80211PacketHaveAttributeWithUint32Value, attr, expected_value,
"Check if the netlink packet has atttribute |attr| with "
"|expected_value|") {
uint32_t actual_value;
if (!arg.GetAttributeValue(attr, &actual_value)) {
return false;
}
return actual_value == expected_value;
}
TEST_F(ScanUtilsTest, CanGetScanResult) {
vector<NativeScanResult> scan_results;
EXPECT_CALL(
netlink_manager_,
SendMessageAndGetResponses(
DoesNL80211PacketMatchCommand(NL80211_CMD_GET_SCAN), _));
// We don't use EXPECT_TRUE here because we need to mock a complete
// response for NL80211_CMD_GET_SCAN to satisfy the parsing code called
// by GetScanResult.
// TODO(b/34231002): Mock response for NL80211_CMD_GET_SCAN.
// TODO(b/34231420): Add validation of interface index.
scan_utils_.GetScanResult(kFakeInterfaceIndex, &scan_results);
}
TEST_F(ScanUtilsTest, CanSendScanRequest) {
NL80211Packet response = CreateControlMessageAck();
EXPECT_CALL(
netlink_manager_,
SendMessageAndGetResponses(
DoesNL80211PacketMatchCommand(NL80211_CMD_TRIGGER_SCAN), _)).
WillOnce(Invoke(bind(
AppendMessageAndReturn, response, true, _1, _2)));
int errno_ignored;
EXPECT_TRUE(scan_utils_.Scan(kFakeInterfaceIndex, kFakeUseRandomMAC,
kFakeScanType, {}, {}, &errno_ignored));
// TODO(b/34231420): Add validation of requested scan ssids, threshold,
// and frequencies.
}
TEST_F(ScanUtilsTest, CanSendScanRequestWithRandomAddr) {
NL80211Packet response = CreateControlMessageAck();
EXPECT_CALL(
netlink_manager_,
SendMessageAndGetResponses(
AllOf(
DoesNL80211PacketMatchCommand(NL80211_CMD_TRIGGER_SCAN),
DoesNL80211PacketHaveAttributeWithUint32Value(
NL80211_ATTR_SCAN_FLAGS, NL80211_SCAN_FLAG_RANDOM_ADDR)),
_)).
WillOnce(Invoke(bind(AppendMessageAndReturn, response, true, _1, _2)));
int errno_ignored;
EXPECT_TRUE(scan_utils_.Scan(kFakeInterfaceIndex, true,
IWifiScannerImpl::SCAN_TYPE_DEFAULT,
{}, {}, &errno_ignored));
}
TEST_F(ScanUtilsTest, CanSendScanRequestForLowSpanScan) {
NL80211Packet response = CreateControlMessageAck();
EXPECT_CALL(
netlink_manager_,
SendMessageAndGetResponses(
AllOf(
DoesNL80211PacketMatchCommand(NL80211_CMD_TRIGGER_SCAN),
DoesNL80211PacketHaveAttributeWithUint32Value(
NL80211_ATTR_SCAN_FLAGS, NL80211_SCAN_FLAG_LOW_SPAN)),
_)).
WillOnce(Invoke(bind(AppendMessageAndReturn, response, true, _1, _2)));
int errno_ignored;
EXPECT_TRUE(scan_utils_.Scan(kFakeInterfaceIndex, false,
IWifiScannerImpl::SCAN_TYPE_LOW_SPAN,
{}, {}, &errno_ignored));
}
TEST_F(ScanUtilsTest, CanSendScanRequestForLowPowerScan) {
NL80211Packet response = CreateControlMessageAck();
EXPECT_CALL(
netlink_manager_,
SendMessageAndGetResponses(
AllOf(
DoesNL80211PacketMatchCommand(NL80211_CMD_TRIGGER_SCAN),
DoesNL80211PacketHaveAttributeWithUint32Value(
NL80211_ATTR_SCAN_FLAGS, NL80211_SCAN_FLAG_LOW_POWER)),
_)).
WillOnce(Invoke(bind(AppendMessageAndReturn, response, true, _1, _2)));
int errno_ignored;
EXPECT_TRUE(scan_utils_.Scan(kFakeInterfaceIndex, false,
IWifiScannerImpl::SCAN_TYPE_LOW_POWER,
{}, {}, &errno_ignored));
}
TEST_F(ScanUtilsTest, CanSendScanRequestForHighAccuracyScan) {
NL80211Packet response = CreateControlMessageAck();
EXPECT_CALL(
netlink_manager_,
SendMessageAndGetResponses(
AllOf(
DoesNL80211PacketMatchCommand(NL80211_CMD_TRIGGER_SCAN),
DoesNL80211PacketHaveAttributeWithUint32Value(
NL80211_ATTR_SCAN_FLAGS, NL80211_SCAN_FLAG_HIGH_ACCURACY)),
_)).
WillOnce(Invoke(bind(AppendMessageAndReturn, response, true, _1, _2)));
int errno_ignored;
EXPECT_TRUE(scan_utils_.Scan(kFakeInterfaceIndex, false,
IWifiScannerImpl::SCAN_TYPE_HIGH_ACCURACY,
{}, {}, &errno_ignored));
}
TEST_F(ScanUtilsTest, CanSendScanRequestForHighAccuracyScanWithRandomAddr) {
NL80211Packet response = CreateControlMessageAck();
EXPECT_CALL(
netlink_manager_,
SendMessageAndGetResponses(
AllOf(
DoesNL80211PacketMatchCommand(NL80211_CMD_TRIGGER_SCAN),
DoesNL80211PacketHaveAttributeWithUint32Value(
NL80211_ATTR_SCAN_FLAGS,
static_cast<uint32_t>(NL80211_SCAN_FLAG_RANDOM_ADDR |
NL80211_SCAN_FLAG_HIGH_ACCURACY))),
_)).
WillOnce(Invoke(bind(AppendMessageAndReturn, response, true, _1, _2)));
int errno_ignored;
EXPECT_TRUE(scan_utils_.Scan(kFakeInterfaceIndex, true,
IWifiScannerImpl::SCAN_TYPE_HIGH_ACCURACY,
{}, {}, &errno_ignored));
}
TEST_F(ScanUtilsTest, CanHandleScanRequestFailure) {
NL80211Packet response = CreateControlMessageError(kFakeErrorCode);
EXPECT_CALL(
netlink_manager_,
SendMessageAndGetResponses(
DoesNL80211PacketMatchCommand(NL80211_CMD_TRIGGER_SCAN), _)).
WillOnce(Invoke(bind(
AppendMessageAndReturn, response, true, _1, _2)));
int error_code;
EXPECT_FALSE(scan_utils_.Scan(kFakeInterfaceIndex, kFakeUseRandomMAC,
kFakeScanType, {}, {}, &error_code));
EXPECT_EQ(kFakeErrorCode, error_code);
}
TEST_F(ScanUtilsTest, CanSendSchedScanRequest) {
NL80211Packet response = CreateControlMessageAck();
EXPECT_CALL(
netlink_manager_,
SendMessageAndGetResponses(
DoesNL80211PacketMatchCommand(NL80211_CMD_START_SCHED_SCAN), _)).
WillOnce(Invoke(bind(
AppendMessageAndReturn, response, true, _1, _2)));
const SchedScanReqFlags req_flags = {
kFakeUseRandomMAC, kFakeRequestLowPower, kFakeRequestSchedScanRelativeRssi
};
int errno_ignored;
EXPECT_TRUE(scan_utils_.StartScheduledScan(
kFakeInterfaceIndex,
SchedScanIntervalSetting(),
kFake2gRssiThreshold, kFake5gRssiThreshold, req_flags, {}, {}, {},
&errno_ignored));
// TODO(b/34231420): Add validation of requested scan ssids, threshold,
// and frequencies.
}
TEST_F(ScanUtilsTest, CanHandleSchedScanRequestFailure) {
NL80211Packet response = CreateControlMessageError(kFakeErrorCode);
EXPECT_CALL(
netlink_manager_,
SendMessageAndGetResponses(
DoesNL80211PacketMatchCommand(NL80211_CMD_START_SCHED_SCAN), _)).
WillOnce(Invoke(bind(
AppendMessageAndReturn, response, true, _1, _2)));
const SchedScanReqFlags req_flags = {
kFakeUseRandomMAC, kFakeRequestLowPower, kFakeRequestSchedScanRelativeRssi
};
int error_code;
EXPECT_FALSE(scan_utils_.StartScheduledScan(
kFakeInterfaceIndex,
SchedScanIntervalSetting(),
kFake2gRssiThreshold, kFake5gRssiThreshold,
req_flags, {}, {}, {}, &error_code));
EXPECT_EQ(kFakeErrorCode, error_code);
}
TEST_F(ScanUtilsTest, CanSendSchedScanRequestForLowPowerScan) {
NL80211Packet response = CreateControlMessageAck();
EXPECT_CALL(
netlink_manager_,
SendMessageAndGetResponses(
AllOf(
DoesNL80211PacketMatchCommand(NL80211_CMD_START_SCHED_SCAN),
DoesNL80211PacketHaveAttributeWithUint32Value(
NL80211_ATTR_SCAN_FLAGS, NL80211_SCAN_FLAG_LOW_POWER)),
_));
int errno_ignored;
const SchedScanReqFlags req_flags = {
false, true, false
};
scan_utils_.StartScheduledScan(
kFakeInterfaceIndex,
SchedScanIntervalSetting(),
kFake2gRssiThreshold, kFake5gRssiThreshold,
req_flags, {}, {}, {}, &errno_ignored);
}
TEST_F(ScanUtilsTest, CanSpecifyScanPlansForSchedScanRequest) {
EXPECT_CALL(
netlink_manager_,
SendMessageAndGetResponses(
AllOf(
DoesNL80211PacketMatchCommand(NL80211_CMD_START_SCHED_SCAN),
DoesNL80211PacketHaveAttribute(NL80211_ATTR_SCHED_SCAN_PLANS),
Not(DoesNL80211PacketHaveAttribute(
NL80211_ATTR_SCHED_SCAN_INTERVAL))),
_));
int errno_ignored;
SchedScanIntervalSetting interval_setting{
{{kFakeScheduledScanIntervalMs, 10 /* repeated times */}},
kFakeScheduledScanIntervalMs * 3 /* interval for infinite scans */};
const SchedScanReqFlags req_flags = {
kFakeUseRandomMAC, kFakeRequestLowPower, kFakeRequestSchedScanRelativeRssi
};
scan_utils_.StartScheduledScan(
kFakeInterfaceIndex,
interval_setting,
kFake2gRssiThreshold, kFake5gRssiThreshold,
req_flags, {}, {}, {}, &errno_ignored);
}
TEST_F(ScanUtilsTest, CanSpecifySingleIntervalForSchedScanRequest) {
EXPECT_CALL(
netlink_manager_,
SendMessageAndGetResponses(
AllOf(
DoesNL80211PacketMatchCommand(NL80211_CMD_START_SCHED_SCAN),
DoesNL80211PacketHaveAttribute(NL80211_ATTR_SCHED_SCAN_INTERVAL),
Not(DoesNL80211PacketHaveAttribute(
NL80211_ATTR_SCHED_SCAN_PLANS))),
_));
int errno_ignored;
SchedScanIntervalSetting interval_setting{{}, kFakeScheduledScanIntervalMs};
const SchedScanReqFlags req_flags = {
kFakeUseRandomMAC, kFakeRequestLowPower, kFakeRequestSchedScanRelativeRssi
};
scan_utils_.StartScheduledScan(
kFakeInterfaceIndex,
interval_setting,
kFake2gRssiThreshold, kFake5gRssiThreshold,
req_flags, {}, {}, {}, &errno_ignored);
}
TEST_F(ScanUtilsTest, CanPrioritizeLastSeenSinceBootNetlinkAttribute) {
constexpr uint64_t kLastSeenTimestampNanoSeconds = 123456;
constexpr uint64_t kBssTsfTimestampMicroSeconds = 654321;
NL80211NestedAttr bss(NL80211_ATTR_BSS);
bss.AddAttribute(
NL80211Attr<uint64_t>(NL80211_BSS_LAST_SEEN_BOOTTIME,
kLastSeenTimestampNanoSeconds));
bss.AddAttribute(
NL80211Attr<uint64_t>(NL80211_BSS_TSF, kBssTsfTimestampMicroSeconds));
uint64_t timestamp_microseconds;
EXPECT_TRUE(scan_utils_.GetBssTimestampForTesting(
bss, ×tamp_microseconds));
EXPECT_EQ(kLastSeenTimestampNanoSeconds/1000, timestamp_microseconds);
}
TEST_F(ScanUtilsTest, CanHandleMissingLastSeenSinceBootNetlinkAttribute) {
constexpr uint64_t kBssTsfTimestampMicroSeconds = 654321;
NL80211NestedAttr bss(NL80211_ATTR_BSS);
bss.AddAttribute(
NL80211Attr<uint64_t>(NL80211_BSS_TSF, kBssTsfTimestampMicroSeconds));
uint64_t timestamp_microseconds;
EXPECT_TRUE(scan_utils_.GetBssTimestampForTesting(
bss, ×tamp_microseconds));
EXPECT_EQ(kBssTsfTimestampMicroSeconds, timestamp_microseconds);
}
// Probe TSF is newer.
TEST_F(ScanUtilsTest, CanPickMostRecentTimestampBetweenBetweenProbeAndBeacon1) {
constexpr uint64_t kBssBeaconTsfTimestampMicroSeconds = 654321;
constexpr uint64_t kBssTsfTimestampMicroSeconds =
kBssBeaconTsfTimestampMicroSeconds + 2000;
NL80211NestedAttr bss(NL80211_ATTR_BSS);
bss.AddAttribute(
NL80211Attr<uint64_t>(NL80211_BSS_BEACON_TSF,
kBssBeaconTsfTimestampMicroSeconds));
bss.AddAttribute(
NL80211Attr<uint64_t>(NL80211_BSS_TSF, kBssTsfTimestampMicroSeconds));
uint64_t timestamp_microseconds;
EXPECT_TRUE(scan_utils_.GetBssTimestampForTesting(
bss, ×tamp_microseconds));
EXPECT_EQ(kBssTsfTimestampMicroSeconds, timestamp_microseconds);
}
// Beacon TSF is newer.
TEST_F(ScanUtilsTest, CanPickMostRecentTimestampBetweenBetweenProbeAndBeacon2) {
constexpr uint64_t kBssTsfTimestampMicroSeconds = 654321;
constexpr uint64_t kBssBeaconTsfTimestampMicroSeconds =
kBssTsfTimestampMicroSeconds + 2000;
NL80211NestedAttr bss(NL80211_ATTR_BSS);
bss.AddAttribute(
NL80211Attr<uint64_t>(NL80211_BSS_BEACON_TSF,
kBssBeaconTsfTimestampMicroSeconds));
bss.AddAttribute(
NL80211Attr<uint64_t>(NL80211_BSS_TSF, kBssTsfTimestampMicroSeconds));
uint64_t timestamp_microseconds;
EXPECT_TRUE(scan_utils_.GetBssTimestampForTesting(
bss, ×tamp_microseconds));
EXPECT_EQ(kBssBeaconTsfTimestampMicroSeconds, timestamp_microseconds);
}
} // namespace wificond
} // namespace android