use crate::ap::{frame_writer, BufferedFrame, Context, TimedEvent};
use crate::device::DeviceOps;
use crate::disconnect::LocallyInitiated;
use crate::error::Error;
use fdf::ArenaStaticBox;
use ieee80211::{MacAddr, MacAddrBytes, Ssid};
use log::warn;
use std::collections::VecDeque;
use wlan_common::append::Append;
use wlan_common::buffer_writer::BufferWriter;
use wlan_common::mac::{self, Aid, AuthAlgorithmNumber, FrameClass, ReasonCode};
use wlan_common::timer::EventId;
use wlan_common::{ie, TimeUnit};
use wlan_statemachine::StateMachine;
use zerocopy::SplitByteSlice;
use {
fidl_fuchsia_wlan_common as fidl_common, fidl_fuchsia_wlan_ieee80211 as fidl_ieee80211,
fidl_fuchsia_wlan_mlme as fidl_mlme, fidl_fuchsia_wlan_softmac as fidl_softmac,
fuchsia_trace as trace, wlan_trace as wtrace,
};
const BSS_MAX_IDLE_PERIOD: u16 = 90;
#[derive(Debug)]
enum PowerSaveState {
Awake,
Dozing {
buffered: VecDeque<BufferedFrame>,
},
}
#[derive(Debug)]
enum State {
Authenticating,
Authenticated,
Associated {
aid: Aid,
eapol_controlled_port: Option<fidl_mlme::ControlledPortState>,
active_timeout_event_id: Option<EventId>,
ps_state: PowerSaveState,
},
Deauthenticated,
}
impl State {
fn max_frame_class(&self) -> FrameClass {
match self {
State::Deauthenticated | State::Authenticating => FrameClass::Class1,
State::Authenticated => FrameClass::Class2,
State::Associated { .. } => FrameClass::Class3,
}
}
}
pub struct RemoteClient {
pub addr: MacAddr,
state: StateMachine<State>,
}
#[derive(Debug)]
pub enum ClientRejection {
NotPermitted,
Unsupported,
NotAuthenticated,
NotAssociated,
ControlledPortClosed,
ParseFailed,
SmeSendError(Error),
WlanSendError(Error),
EthSendError(Error),
DeviceError(Error),
}
impl ClientRejection {
pub fn log_level(&self) -> log::Level {
match self {
Self::ParseFailed
| Self::SmeSendError(..)
| Self::WlanSendError(..)
| Self::EthSendError(..) => log::Level::Error,
Self::ControlledPortClosed | Self::Unsupported => log::Level::Warn,
_ => log::Level::Trace,
}
}
}
#[derive(Debug)]
pub enum ClientEvent {
BssIdleTimeout,
}
impl RemoteClient {
pub fn new(addr: MacAddr) -> Self {
Self { addr, state: StateMachine::new(State::Authenticating) }
}
pub fn deauthenticated(&self) -> bool {
match self.state.as_ref() {
State::Deauthenticated => true,
_ => false,
}
}
pub fn aid(&self) -> Option<Aid> {
match self.state.as_ref() {
State::Associated { aid, .. } => Some(*aid),
_ => None,
}
}
pub fn has_buffered_frames(&self) -> bool {
match self.state.as_ref() {
State::Associated { ps_state: PowerSaveState::Dozing { buffered }, .. } => {
!buffered.is_empty()
}
_ => false,
}
}
pub fn dozing(&self) -> bool {
match self.state.as_ref() {
State::Associated { ps_state: PowerSaveState::Dozing { .. }, .. } => true,
_ => false,
}
}
async fn change_state<D: DeviceOps>(
&mut self,
ctx: &mut Context<D>,
next_state: State,
) -> Result<(), Error> {
match self.state.as_mut() {
State::Associated { .. } => {
ctx.device
.clear_association(&fidl_softmac::WlanSoftmacBaseClearAssociationRequest {
peer_addr: Some(self.addr.to_array()),
..Default::default()
})
.await
.map_err(|s| Error::Status(format!("failed to clear association"), s))?;
}
_ => (),
}
self.state.replace_state_with(next_state);
Ok(())
}
fn schedule_after<D>(
&self,
ctx: &mut Context<D>,
duration: zx::MonotonicDuration,
event: ClientEvent,
) -> EventId {
ctx.schedule_after(duration, TimedEvent::ClientEvent(self.addr, event))
}
fn schedule_bss_idle_timeout<D>(&self, ctx: &mut Context<D>) -> EventId {
self.schedule_after(
ctx,
zx::MonotonicDuration::from(TimeUnit(1000)) * (BSS_MAX_IDLE_PERIOD as i64),
ClientEvent::BssIdleTimeout,
)
}
async fn handle_bss_idle_timeout<D: DeviceOps>(
&mut self,
ctx: &mut Context<D>,
event_id: EventId,
) -> Result<(), ClientRejection> {
match self.state.as_ref() {
State::Associated { active_timeout_event_id, .. } => {
if *active_timeout_event_id != Some(event_id) {
return Ok(());
}
}
_ => {
return Ok(());
}
}
self.change_state(ctx, State::Authenticated).await.map_err(ClientRejection::DeviceError)?;
let buffer = ctx
.make_disassoc_frame(
self.addr.clone(),
fidl_ieee80211::ReasonCode::ReasonInactivity.into(),
)
.map_err(ClientRejection::WlanSendError)?;
self.send_wlan_frame(ctx, buffer, fidl_softmac::WlanTxInfoFlags::empty(), None).map_err(
|s| {
ClientRejection::WlanSendError(Error::Status(
format!("error sending disassoc frame on BSS idle timeout"),
s,
))
},
)?;
ctx.send_mlme_disassoc_ind(
self.addr.clone(),
fidl_ieee80211::ReasonCode::ReasonInactivity,
LocallyInitiated(true),
)
.map_err(ClientRejection::SmeSendError)?;
Ok(())
}
fn reset_bss_max_idle_timeout<D>(&mut self, ctx: &mut Context<D>) {
let new_active_timeout_event_id = match self.state.as_ref() {
State::Associated { .. } => Some(self.schedule_bss_idle_timeout(ctx)),
_ => None,
};
match self.state.as_mut() {
State::Associated { active_timeout_event_id, .. } => {
*active_timeout_event_id = new_active_timeout_event_id;
}
_ => (),
}
}
fn is_frame_class_permitted(&self, frame_class: FrameClass) -> bool {
frame_class <= self.state.as_ref().max_frame_class()
}
pub async fn handle_event<D: DeviceOps>(
&mut self,
ctx: &mut Context<D>,
event_id: EventId,
event: ClientEvent,
) -> Result<(), ClientRejection> {
match event {
ClientEvent::BssIdleTimeout => self.handle_bss_idle_timeout(ctx, event_id).await,
}
}
pub async fn handle_mlme_auth_resp<D: DeviceOps>(
&mut self,
ctx: &mut Context<D>,
result_code: fidl_mlme::AuthenticateResultCode,
) -> Result<(), Error> {
log::info!("enter handle_mlme_auth_resp");
self.change_state(
ctx,
if result_code == fidl_mlme::AuthenticateResultCode::Success {
State::Authenticated
} else {
State::Deauthenticated
},
)
.await?;
log::info!("creating auth frame");
let buffer = ctx.make_auth_frame(
self.addr.clone(),
AuthAlgorithmNumber::OPEN,
2,
match result_code {
fidl_mlme::AuthenticateResultCode::Success => {
fidl_ieee80211::StatusCode::Success.into()
}
fidl_mlme::AuthenticateResultCode::Refused => {
fidl_ieee80211::StatusCode::RefusedReasonUnspecified.into()
}
fidl_mlme::AuthenticateResultCode::AntiCloggingTokenRequired => {
fidl_ieee80211::StatusCode::AntiCloggingTokenRequired.into()
}
fidl_mlme::AuthenticateResultCode::FiniteCyclicGroupNotSupported => {
fidl_ieee80211::StatusCode::UnsupportedFiniteCyclicGroup.into()
}
fidl_mlme::AuthenticateResultCode::AuthenticationRejected => {
fidl_ieee80211::StatusCode::ChallengeFailure.into()
}
fidl_mlme::AuthenticateResultCode::AuthFailureTimeout => {
fidl_ieee80211::StatusCode::RejectedSequenceTimeout.into()
}
},
)?;
log::info!("Sending auth frame to driver: {} bytes", buffer.len());
self.send_wlan_frame(ctx, buffer, fidl_softmac::WlanTxInfoFlags::empty(), None)
.map_err(|s| Error::Status(format!("error sending auth frame"), s))
}
pub async fn handle_mlme_deauth_req<D: DeviceOps>(
&mut self,
ctx: &mut Context<D>,
reason_code: fidl_ieee80211::ReasonCode,
) -> Result<(), Error> {
self.change_state(ctx, State::Deauthenticated).await?;
let buffer = ctx.make_deauth_frame(self.addr.clone(), reason_code.into())?;
self.send_wlan_frame(ctx, buffer, fidl_softmac::WlanTxInfoFlags::empty(), None)
.map_err(|s| Error::Status(format!("error sending deauth frame"), s))
}
pub async fn handle_mlme_assoc_resp<D: DeviceOps>(
&mut self,
ctx: &mut Context<D>,
is_rsn: bool,
channel: u8,
capabilities: mac::CapabilityInfo,
result_code: fidl_mlme::AssociateResultCode,
aid: Aid,
rates: &[u8],
) -> Result<(), Error> {
self.change_state(
ctx,
if result_code == fidl_mlme::AssociateResultCode::Success {
State::Associated {
aid,
eapol_controlled_port: if is_rsn {
Some(fidl_mlme::ControlledPortState::Closed)
} else {
None
},
active_timeout_event_id: None,
ps_state: PowerSaveState::Awake,
}
} else {
State::Authenticated
},
)
.await?;
if let State::Associated { .. } = self.state.as_ref() {
self.reset_bss_max_idle_timeout(ctx);
ctx.device
.notify_association_complete(fidl_softmac::WlanAssociationConfig {
bssid: Some(self.addr.to_array()),
aid: Some(aid),
listen_interval: None, channel: Some(fidl_common::WlanChannel {
primary: channel,
cbw: fidl_common::ChannelBandwidth::Cbw20,
secondary80: 0,
}),
qos: Some(false),
wmm_params: None,
rates: Some(rates.to_vec()),
capability_info: Some(capabilities.raw()),
ht_cap: None,
ht_op: None,
vht_cap: None,
vht_op: None,
..Default::default()
})
.await
.map_err(|s| Error::Status(format!("failed to configure association"), s))?;
}
let buffer = match result_code {
fidl_mlme::AssociateResultCode::Success => ctx.make_assoc_resp_frame(
self.addr,
capabilities,
aid,
rates,
Some(BSS_MAX_IDLE_PERIOD),
),
_ => ctx.make_assoc_resp_frame_error(
self.addr,
capabilities,
match result_code {
fidl_mlme::AssociateResultCode::Success => {
panic!("Success should have already been handled");
}
fidl_mlme::AssociateResultCode::RefusedReasonUnspecified => {
fidl_ieee80211::StatusCode::RefusedReasonUnspecified.into()
}
fidl_mlme::AssociateResultCode::RefusedNotAuthenticated => {
fidl_ieee80211::StatusCode::RefusedUnauthenticatedAccessNotSupported.into()
}
fidl_mlme::AssociateResultCode::RefusedCapabilitiesMismatch => {
fidl_ieee80211::StatusCode::RefusedCapabilitiesMismatch.into()
}
fidl_mlme::AssociateResultCode::RefusedExternalReason => {
fidl_ieee80211::StatusCode::RefusedExternalReason.into()
}
fidl_mlme::AssociateResultCode::RefusedApOutOfMemory => {
fidl_ieee80211::StatusCode::RefusedApOutOfMemory.into()
}
fidl_mlme::AssociateResultCode::RefusedBasicRatesMismatch => {
fidl_ieee80211::StatusCode::RefusedBasicRatesMismatch.into()
}
fidl_mlme::AssociateResultCode::RejectedEmergencyServicesNotSupported => {
fidl_ieee80211::StatusCode::RejectedEmergencyServicesNotSupported.into()
}
fidl_mlme::AssociateResultCode::RefusedTemporarily => {
fidl_ieee80211::StatusCode::RefusedTemporarily.into()
}
},
),
}?;
self.send_wlan_frame(ctx, buffer, fidl_softmac::WlanTxInfoFlags::empty(), None)
.map_err(|s| Error::Status(format!("error sending assoc frame"), s))
}
pub async fn handle_mlme_disassoc_req<D: DeviceOps>(
&mut self,
ctx: &mut Context<D>,
reason_code: u16,
) -> Result<(), Error> {
self.change_state(ctx, State::Authenticated).await?;
let buffer = ctx.make_disassoc_frame(self.addr.clone(), ReasonCode(reason_code))?;
self.send_wlan_frame(ctx, buffer, fidl_softmac::WlanTxInfoFlags::empty(), None)
.map_err(|s| Error::Status(format!("error sending disassoc frame"), s))
}
pub fn handle_mlme_set_controlled_port_req(
&mut self,
state: fidl_mlme::ControlledPortState,
) -> Result<(), Error> {
match self.state.as_mut() {
State::Associated {
eapol_controlled_port: eapol_controlled_port @ Some(_), ..
} => {
eapol_controlled_port.replace(state);
Ok(())
}
State::Associated { eapol_controlled_port: None, .. } => {
Err(Error::Status(format!("client is not in an RSN"), zx::Status::BAD_STATE))
}
_ => Err(Error::Status(format!("client is not associated"), zx::Status::BAD_STATE)),
}
}
pub fn handle_mlme_eapol_req<D: DeviceOps>(
&mut self,
ctx: &mut Context<D>,
src_addr: MacAddr,
data: &[u8],
) -> Result<(), Error> {
let buffer = ctx.make_eapol_frame(self.addr, src_addr, false, data)?;
self.send_wlan_frame(ctx, buffer, fidl_softmac::WlanTxInfoFlags::FAVOR_RELIABILITY, None)
.map_err(|s| Error::Status(format!("error sending eapol frame"), s))
}
async fn handle_disassoc_frame<D: DeviceOps>(
&mut self,
ctx: &mut Context<D>,
reason_code: ReasonCode,
) -> Result<(), ClientRejection> {
self.change_state(ctx, State::Authenticated).await.map_err(ClientRejection::DeviceError)?;
ctx.send_mlme_disassoc_ind(
self.addr.clone(),
Option::<fidl_ieee80211::ReasonCode>::from(reason_code)
.unwrap_or(fidl_ieee80211::ReasonCode::UnspecifiedReason),
LocallyInitiated(false),
)
.map_err(ClientRejection::SmeSendError)
}
fn handle_assoc_req_frame<D: DeviceOps>(
&self,
ctx: &mut Context<D>,
capabilities: mac::CapabilityInfo,
listen_interval: u16,
ssid: Option<Ssid>,
rates: Vec<ie::SupportedRate>,
rsne: Option<Vec<u8>>,
) -> Result<(), ClientRejection> {
ctx.send_mlme_assoc_ind(self.addr.clone(), listen_interval, ssid, capabilities, rates, rsne)
.map_err(ClientRejection::SmeSendError)
}
async fn handle_auth_frame<D: DeviceOps>(
&mut self,
ctx: &mut Context<D>,
auth_alg_num: AuthAlgorithmNumber,
) -> Result<(), ClientRejection> {
ctx.send_mlme_auth_ind(
self.addr.clone(),
match auth_alg_num {
AuthAlgorithmNumber::OPEN => fidl_mlme::AuthenticationTypes::OpenSystem,
AuthAlgorithmNumber::SHARED_KEY => fidl_mlme::AuthenticationTypes::SharedKey,
AuthAlgorithmNumber::FAST_BSS_TRANSITION => {
fidl_mlme::AuthenticationTypes::FastBssTransition
}
AuthAlgorithmNumber::SAE => fidl_mlme::AuthenticationTypes::Sae,
_ => {
self.change_state(ctx, State::Deauthenticated)
.await
.map_err(ClientRejection::DeviceError)?;
let buffer = ctx
.make_auth_frame(
self.addr.clone(),
auth_alg_num,
2,
fidl_ieee80211::StatusCode::UnsupportedAuthAlgorithm.into(),
)
.map_err(ClientRejection::WlanSendError)?;
return self
.send_wlan_frame(ctx, buffer, fidl_softmac::WlanTxInfoFlags::empty(), None)
.map_err(|s| {
ClientRejection::WlanSendError(Error::Status(
format!("failed to send auth frame"),
s,
))
});
}
},
)
.map_err(ClientRejection::SmeSendError)
}
async fn handle_deauth_frame<D: DeviceOps>(
&mut self,
ctx: &mut Context<D>,
reason_code: ReasonCode,
) -> Result<(), ClientRejection> {
self.change_state(ctx, State::Deauthenticated)
.await
.map_err(ClientRejection::DeviceError)?;
ctx.send_mlme_deauth_ind(
self.addr.clone(),
Option::<fidl_ieee80211::ReasonCode>::from(reason_code)
.unwrap_or(fidl_ieee80211::ReasonCode::UnspecifiedReason),
LocallyInitiated(false),
)
.map_err(ClientRejection::SmeSendError)
}
fn handle_action_frame<D>(&self, _ctx: &mut Context<D>) -> Result<(), ClientRejection> {
Ok(())
}
pub fn handle_ps_poll<D: DeviceOps>(
&mut self,
ctx: &mut Context<D>,
aid: Aid,
) -> Result<(), ClientRejection> {
self.reject_frame_class_if_not_permitted(ctx, mac::FrameClass::Class3)?;
match self.state.as_mut() {
State::Associated { aid: current_aid, ps_state, .. } => {
if aid != *current_aid {
return Err(ClientRejection::NotPermitted);
}
match ps_state {
PowerSaveState::Dozing { buffered } => {
let BufferedFrame { mut buffer, tx_flags, async_id } =
match buffered.pop_front() {
Some(buffered) => buffered,
None => {
return Ok(());
}
};
if !buffered.is_empty() {
frame_writer::set_more_data(&mut buffer[..]).map_err(|e| {
wtrace::async_end_wlansoftmac_tx(async_id, zx::Status::INTERNAL);
ClientRejection::WlanSendError(e)
})?;
}
ctx.device.send_wlan_frame(buffer, tx_flags, None).map_err(|s| {
wtrace::async_end_wlansoftmac_tx(async_id, s);
ClientRejection::WlanSendError(Error::Status(
format!("error sending buffered frame on PS-Poll"),
s,
))
})?;
}
_ => {
return Err(ClientRejection::NotPermitted);
}
}
}
_ => {
return Err(ClientRejection::NotAssociated);
}
};
Ok(())
}
fn doze(&mut self) -> Result<(), ClientRejection> {
match self.state.as_mut() {
State::Associated { ps_state, .. } => match ps_state {
PowerSaveState::Awake => {
*ps_state = PowerSaveState::Dozing {
buffered: VecDeque::new(),
}
}
PowerSaveState::Dozing { .. } => {}
},
_ => {
return Err(ClientRejection::NotAssociated);
}
};
Ok(())
}
fn wake<D: DeviceOps>(&mut self, ctx: &mut Context<D>) -> Result<(), ClientRejection> {
match self.state.as_mut() {
State::Associated { ps_state, .. } => {
let mut old_ps_state = PowerSaveState::Awake;
std::mem::swap(ps_state, &mut old_ps_state);
let mut buffered = match old_ps_state {
PowerSaveState::Awake => {
return Ok(());
}
PowerSaveState::Dozing { buffered } => buffered.into_iter().peekable(),
};
while let Some(BufferedFrame { mut buffer, tx_flags, async_id }) = buffered.next() {
if buffered.peek().is_some() {
frame_writer::set_more_data(&mut buffer[..])
.map_err(ClientRejection::WlanSendError)?;
}
ctx.device.send_wlan_frame(buffer, tx_flags, Some(async_id)).map_err(|s| {
ClientRejection::WlanSendError(Error::Status(
format!("error sending buffered frame on wake"),
s,
))
})?;
}
}
_ => {
return Ok(());
}
};
Ok(())
}
pub fn set_power_state<D: DeviceOps>(
&mut self,
ctx: &mut Context<D>,
power_state: mac::PowerState,
) -> Result<(), ClientRejection> {
match power_state {
mac::PowerState::AWAKE => self.wake(ctx),
mac::PowerState::DOZE => self.doze(),
}
}
fn handle_eapol_llc_frame<D: DeviceOps>(
&self,
ctx: &mut Context<D>,
dst_addr: MacAddr,
src_addr: MacAddr,
body: &[u8],
) -> Result<(), ClientRejection> {
ctx.send_mlme_eapol_ind(dst_addr, src_addr, &body).map_err(ClientRejection::SmeSendError)
}
fn handle_llc_frame<D: DeviceOps>(
&self,
ctx: &mut Context<D>,
dst_addr: MacAddr,
src_addr: MacAddr,
ether_type: u16,
body: &[u8],
) -> Result<(), ClientRejection> {
ctx.deliver_eth_frame(dst_addr, src_addr, ether_type, body)
.map_err(ClientRejection::EthSendError)
}
fn reject_frame_class_if_not_permitted<D: DeviceOps>(
&mut self,
ctx: &mut Context<D>,
frame_class: FrameClass,
) -> Result<(), ClientRejection> {
if self.is_frame_class_permitted(frame_class) {
return Ok(());
}
let reason_code = match frame_class {
FrameClass::Class1 => panic!("class 1 frames should always be permitted"),
FrameClass::Class2 => fidl_ieee80211::ReasonCode::InvalidClass2Frame,
FrameClass::Class3 => fidl_ieee80211::ReasonCode::InvalidClass3Frame,
};
match self.state.as_ref() {
State::Deauthenticated | State::Authenticating => {
let buffer = ctx
.make_deauth_frame(self.addr, reason_code.into())
.map_err(ClientRejection::WlanSendError)?;
self.send_wlan_frame(ctx, buffer, fidl_softmac::WlanTxInfoFlags::empty(), None)
.map_err(|s| {
ClientRejection::WlanSendError(Error::Status(
format!("failed to send deauth frame"),
s,
))
})?;
ctx.send_mlme_deauth_ind(self.addr, reason_code, LocallyInitiated(true))
.map_err(ClientRejection::SmeSendError)?;
}
State::Authenticated => {
let buffer = ctx
.make_disassoc_frame(self.addr, reason_code.into())
.map_err(ClientRejection::WlanSendError)?;
self.send_wlan_frame(ctx, buffer, fidl_softmac::WlanTxInfoFlags::empty(), None)
.map_err(|s| {
ClientRejection::WlanSendError(Error::Status(
format!("failed to send disassoc frame"),
s,
))
})?;
ctx.send_mlme_disassoc_ind(self.addr, reason_code, LocallyInitiated(true))
.map_err(ClientRejection::SmeSendError)?;
}
State::Associated { .. } => {
panic!("all frames should be permitted for an associated client")
}
};
return Err(ClientRejection::NotPermitted);
}
pub async fn handle_mgmt_frame<B: SplitByteSlice, D: DeviceOps>(
&mut self,
ctx: &mut Context<D>,
capabilities: mac::CapabilityInfo,
ssid: Option<Ssid>,
mgmt_frame: mac::MgmtFrame<B>,
) -> Result<(), ClientRejection> {
self.reject_frame_class_if_not_permitted(ctx, mac::frame_class(&mgmt_frame.frame_ctrl()))?;
self.reset_bss_max_idle_timeout(ctx);
match mgmt_frame.try_into_mgmt_body().1.ok_or(ClientRejection::ParseFailed)? {
mac::MgmtBody::Authentication(mac::AuthFrame { auth_hdr, .. }) => {
self.handle_auth_frame(ctx, auth_hdr.auth_alg_num).await
}
mac::MgmtBody::AssociationReq(assoc_req_frame) => {
let mut rates = vec![];
let mut rsne = None;
for (id, ie_body) in assoc_req_frame.ies() {
match id {
ie::Id::SUPPORTED_RATES => {
match ie::parse_supported_rates(ie_body) {
Ok(supported_rates) => rates.extend(&*supported_rates),
Err(e) => warn!("{:?}", e),
};
}
ie::Id::EXTENDED_SUPPORTED_RATES => {
match ie::parse_extended_supported_rates(ie_body) {
Ok(extended_supported_rates) => {
rates.extend(&*extended_supported_rates)
}
Err(e) => warn!("{:?}", e),
};
}
ie::Id::RSNE => {
rsne = Some({
let mut buffer =
vec![0; std::mem::size_of::<ie::Header>() + ie_body.len()];
let mut w = BufferWriter::new(&mut buffer[..]);
w.append_value(&ie::Header {
id: ie::Id::RSNE,
body_len: ie_body.len() as u8,
})
.expect("expected enough room in buffer for IE header");
w.append_bytes(ie_body)
.expect("expected enough room in buffer for IE body");
buffer
});
}
_ => {}
}
}
self.handle_assoc_req_frame(
ctx,
capabilities,
assoc_req_frame.assoc_req_hdr.listen_interval,
ssid,
rates,
rsne,
)
}
mac::MgmtBody::Deauthentication { deauth_hdr, .. } => {
self.handle_deauth_frame(ctx, deauth_hdr.reason_code).await
}
mac::MgmtBody::Disassociation { disassoc_hdr, .. } => {
self.handle_disassoc_frame(ctx, disassoc_hdr.reason_code).await
}
mac::MgmtBody::Action(_) => self.handle_action_frame(ctx),
_ => Err(ClientRejection::Unsupported),
}
}
pub fn handle_data_frame<B: SplitByteSlice, D: DeviceOps>(
&mut self,
ctx: &mut Context<D>,
data_frame: mac::DataFrame<B>,
) -> Result<(), ClientRejection> {
self.reject_frame_class_if_not_permitted(ctx, mac::frame_class(&data_frame.frame_ctrl()))?;
self.reset_bss_max_idle_timeout(ctx);
for msdu in data_frame {
let mac::Msdu { dst_addr, src_addr, llc_frame } = msdu;
match llc_frame.hdr.protocol_id.to_native() {
mac::ETHER_TYPE_EAPOL => {
self.handle_eapol_llc_frame(ctx, dst_addr, src_addr, &llc_frame.body)?
}
_ => match self.state.as_ref() {
State::Associated {
eapol_controlled_port: Some(fidl_mlme::ControlledPortState::Closed),
..
} => (),
_ => self.handle_llc_frame(
ctx,
dst_addr,
src_addr,
llc_frame.hdr.protocol_id.to_native(),
&llc_frame.body,
)?,
},
}
}
Ok(())
}
pub fn handle_eth_frame<D: DeviceOps>(
&mut self,
ctx: &mut Context<D>,
dst_addr: MacAddr,
src_addr: MacAddr,
ether_type: u16,
body: &[u8],
async_id: trace::Id,
) -> Result<(), ClientRejection> {
let eapol_controlled_port = match self.state.as_ref() {
State::Associated { eapol_controlled_port, .. } => eapol_controlled_port,
_ => {
return Err(ClientRejection::NotAssociated);
}
};
let protection = match eapol_controlled_port {
None => false,
Some(fidl_mlme::ControlledPortState::Open) => true,
Some(fidl_mlme::ControlledPortState::Closed) => {
return Err(ClientRejection::ControlledPortClosed);
}
};
let buffer = ctx
.make_data_frame(
dst_addr, src_addr, protection,
false, ether_type, body,
)
.map_err(ClientRejection::WlanSendError)?;
self.send_wlan_frame(ctx, buffer, fidl_softmac::WlanTxInfoFlags::empty(), Some(async_id))
.map_err(move |s| {
ClientRejection::WlanSendError(Error::Status(
format!("error sending eapol frame"),
s,
))
})
}
pub fn send_wlan_frame<D: DeviceOps>(
&mut self,
ctx: &mut Context<D>,
buffer: ArenaStaticBox<[u8]>,
tx_flags: fidl_softmac::WlanTxInfoFlags,
async_id: Option<trace::Id>,
) -> Result<(), zx::Status> {
let async_id = async_id.unwrap_or_else(|| {
let async_id = trace::Id::new();
wtrace::async_begin_wlansoftmac_tx(async_id, "mlme");
async_id
});
match self.state.as_mut() {
State::Associated { ps_state, .. } => match ps_state {
PowerSaveState::Awake => {
ctx.device.send_wlan_frame(buffer, tx_flags, Some(async_id))
}
PowerSaveState::Dozing { buffered } => {
buffered.push_back(BufferedFrame { buffer, tx_flags, async_id });
Ok(())
}
},
_ => ctx.device.send_wlan_frame(buffer, tx_flags, Some(async_id)),
}
}
}
#[cfg(test)]
mod tests {
use super::*;
use crate::device::FakeDevice;
use ieee80211::Bssid;
use lazy_static::lazy_static;
use test_case::test_case;
use wlan_common::assert_variant;
use wlan_common::mac::{CapabilityInfo, IntoBytesExt as _};
use wlan_common::test_utils::fake_frames::*;
use wlan_common::timer::{self, create_timer};
use zerocopy::Unalign;
lazy_static! {
static ref CLIENT_ADDR: MacAddr = [1; 6].into();
static ref AP_ADDR: Bssid = [2; 6].into();
static ref CLIENT_ADDR2: MacAddr = [3; 6].into();
}
fn make_remote_client() -> RemoteClient {
RemoteClient::new(*CLIENT_ADDR)
}
fn make_context(
fake_device: FakeDevice,
) -> (Context<FakeDevice>, timer::EventStream<TimedEvent>) {
let (timer, time_stream) = create_timer();
(Context::new(fake_device, timer, *AP_ADDR), time_stream)
}
#[fuchsia::test(allow_stalls = false)]
async fn handle_mlme_auth_resp() {
let (fake_device, fake_device_state) = FakeDevice::new().await;
let mut r_sta = make_remote_client();
let (mut ctx, _) = make_context(fake_device);
r_sta
.handle_mlme_auth_resp(&mut ctx, fidl_mlme::AuthenticateResultCode::Success)
.await
.expect("expected OK");
assert_variant!(r_sta.state.as_ref(), State::Authenticated);
assert_eq!(fake_device_state.lock().wlan_queue.len(), 1);
#[rustfmt::skip]
assert_eq!(&fake_device_state.lock().wlan_queue[0].0[..], &[
0b10110000, 0, 0, 0, 1, 1, 1, 1, 1, 1, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 0x10, 0, 0, 0, 2, 0, 0, 0, ][..]);
}
#[fuchsia::test(allow_stalls = false)]
async fn handle_mlme_auth_resp_failure() {
let (fake_device, fake_device_state) = FakeDevice::new().await;
let mut r_sta = make_remote_client();
let (mut ctx, _) = make_context(fake_device);
r_sta
.handle_mlme_auth_resp(
&mut ctx,
fidl_mlme::AuthenticateResultCode::AntiCloggingTokenRequired,
)
.await
.expect("expected OK");
assert_variant!(r_sta.state.as_ref(), State::Deauthenticated);
assert_eq!(fake_device_state.lock().wlan_queue.len(), 1);
#[rustfmt::skip]
assert_eq!(&fake_device_state.lock().wlan_queue[0].0[..], &[
0b10110000, 0, 0, 0, 1, 1, 1, 1, 1, 1, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 0x10, 0, 0, 0, 2, 0, 76, 0, ][..]);
}
#[test_case(State::Authenticating; "in authenticating state")]
#[test_case(State::Authenticated; "in authenticated state")]
#[test_case(State::Associated {
aid: 1,
eapol_controlled_port: None,
active_timeout_event_id: None,
ps_state: PowerSaveState::Awake,
}; "in associated state")]
#[fuchsia::test(allow_stalls = false)]
async fn handle_mlme_deauth_req(init_state: State) {
let (fake_device, fake_device_state) = FakeDevice::new().await;
let mut r_sta = make_remote_client();
r_sta.state = StateMachine::new(init_state);
let (mut ctx, _) = make_context(fake_device);
r_sta
.handle_mlme_deauth_req(&mut ctx, fidl_ieee80211::ReasonCode::LeavingNetworkDeauth)
.await
.expect("expected OK");
assert_variant!(r_sta.state.as_ref(), State::Deauthenticated);
assert_eq!(fake_device_state.lock().wlan_queue.len(), 1);
#[rustfmt::skip]
assert_eq!(&fake_device_state.lock().wlan_queue[0].0[..], &[
0b11000000, 0, 0, 0, 1, 1, 1, 1, 1, 1, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 0x10, 0, 3, 0, ][..]);
}
#[fuchsia::test(allow_stalls = false)]
async fn handle_mlme_assoc_resp() {
let (fake_device, fake_device_state) = FakeDevice::new().await;
let mut r_sta = make_remote_client();
let (mut ctx, mut time_stream) = make_context(fake_device);
r_sta
.handle_mlme_assoc_resp(
&mut ctx,
true,
1,
CapabilityInfo(0),
fidl_mlme::AssociateResultCode::Success,
1,
&[1, 2, 3, 4, 5, 6, 7, 8, 9, 10][..],
)
.await
.expect("expected OK");
assert_variant!(
r_sta.state.as_ref(),
State::Associated {
eapol_controlled_port: Some(fidl_mlme::ControlledPortState::Closed),
..
}
);
assert_variant!(r_sta.aid(), Some(aid) => {
assert_eq!(aid, 1);
});
let active_timeout_event_id = match r_sta.state.as_ref() {
State::Associated {
active_timeout_event_id: Some(active_timeout_event_id), ..
} => active_timeout_event_id,
_ => panic!("no active timeout?"),
};
assert_eq!(fake_device_state.lock().wlan_queue.len(), 1);
#[rustfmt::skip]
assert_eq!(&fake_device_state.lock().wlan_queue[0].0[..], &[
0b00010000, 0, 0, 0, 1, 1, 1, 1, 1, 1, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 0x10, 0, 0, 0, 0, 0, 1, 0, 1, 8, 1, 2, 3, 4, 5, 6, 7, 8, 50, 2, 9, 10, 90, 3, 90, 0, 0, ][..]);
let (_, timed_event) =
time_stream.try_next().unwrap().expect("Should have scheduled a timeout");
assert_eq!(timed_event.id, *active_timeout_event_id);
assert!(fake_device_state.lock().assocs.contains_key(&CLIENT_ADDR));
}
#[fuchsia::test(allow_stalls = false)]
async fn handle_mlme_assoc_resp_then_handle_mlme_disassoc_req() {
let (fake_device, fake_device_state) = FakeDevice::new().await;
let mut r_sta = make_remote_client();
let (mut ctx, _) = make_context(fake_device);
r_sta
.handle_mlme_assoc_resp(
&mut ctx,
true,
1,
CapabilityInfo(0),
fidl_mlme::AssociateResultCode::Success,
1,
&[1, 2, 3, 4, 5, 6, 7, 8, 9, 10][..],
)
.await
.expect("expected OK");
assert!(fake_device_state.lock().assocs.contains_key(&CLIENT_ADDR));
r_sta
.handle_mlme_disassoc_req(
&mut ctx,
fidl_ieee80211::ReasonCode::LeavingNetworkDisassoc.into_primitive(),
)
.await
.expect("expected OK");
assert!(!fake_device_state.lock().assocs.contains_key(&CLIENT_ADDR));
}
#[fuchsia::test(allow_stalls = false)]
async fn handle_mlme_assoc_resp_then_handle_mlme_deauth_req() {
let (fake_device, fake_device_state) = FakeDevice::new().await;
let mut r_sta = make_remote_client();
let (mut ctx, _) = make_context(fake_device);
r_sta
.handle_mlme_assoc_resp(
&mut ctx,
true,
1,
CapabilityInfo(0),
fidl_mlme::AssociateResultCode::Success,
1,
&[1, 2, 3, 4, 5, 6, 7, 8, 9, 10][..],
)
.await
.expect("expected OK");
assert!(fake_device_state.lock().assocs.contains_key(&CLIENT_ADDR));
r_sta
.handle_mlme_deauth_req(&mut ctx, fidl_ieee80211::ReasonCode::LeavingNetworkDeauth)
.await
.expect("expected OK");
assert!(!fake_device_state.lock().assocs.contains_key(&CLIENT_ADDR));
}
#[fuchsia::test(allow_stalls = false)]
async fn handle_mlme_assoc_resp_no_rsn() {
let (fake_device, _) = FakeDevice::new().await;
let mut r_sta = make_remote_client();
let (mut ctx, _) = make_context(fake_device);
r_sta
.handle_mlme_assoc_resp(
&mut ctx,
false,
1,
CapabilityInfo(0),
fidl_mlme::AssociateResultCode::Success,
1,
&[1, 2, 3, 4, 5, 6, 7, 8, 9, 10][..],
)
.await
.expect("expected OK");
assert_variant!(
r_sta.state.as_ref(),
State::Associated { eapol_controlled_port: None, active_timeout_event_id: Some(_), .. }
);
}
#[fuchsia::test(allow_stalls = false)]
async fn handle_mlme_assoc_resp_failure_reason_unspecified() {
let (fake_device, fake_device_state) = FakeDevice::new().await;
let mut r_sta = make_remote_client();
let (mut ctx, _) = make_context(fake_device);
r_sta
.handle_mlme_assoc_resp(
&mut ctx,
false,
1,
CapabilityInfo(0),
fidl_mlme::AssociateResultCode::RefusedReasonUnspecified,
1, &[][..],
)
.await
.expect("expected OK");
assert_variant!(r_sta.state.as_ref(), State::Authenticated);
assert_eq!(fake_device_state.lock().wlan_queue.len(), 1);
#[rustfmt::skip]
assert_eq!(&fake_device_state.lock().wlan_queue[0].0[..], &[
0b00010000, 0, 0, 0, 1, 1, 1, 1, 1, 1, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 0x10, 0, 0, 0, 1, 0, 0, 0, ][..]);
}
#[fuchsia::test(allow_stalls = false)]
async fn handle_mlme_assoc_resp_failure_emergency_services_not_supported() {
let (fake_device, fake_device_state) = FakeDevice::new().await;
let mut r_sta = make_remote_client();
let (mut ctx, _) = make_context(fake_device);
r_sta
.handle_mlme_assoc_resp(
&mut ctx,
false,
1,
CapabilityInfo(0),
fidl_mlme::AssociateResultCode::RejectedEmergencyServicesNotSupported,
1, &[][..],
)
.await
.expect("expected OK");
assert_variant!(r_sta.state.as_ref(), State::Authenticated);
assert_eq!(fake_device_state.lock().wlan_queue.len(), 1);
#[rustfmt::skip]
assert_eq!(&fake_device_state.lock().wlan_queue[0].0[..], &[
0b00010000, 0, 0, 0, 1, 1, 1, 1, 1, 1, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 0x10, 0, 0, 0, 94, 0, 0, 0, ][..]);
}
#[fuchsia::test(allow_stalls = false)]
async fn handle_mlme_disassoc_req() {
let (fake_device, fake_device_state) = FakeDevice::new().await;
let mut r_sta = make_remote_client();
let (mut ctx, _) = make_context(fake_device);
r_sta
.handle_mlme_disassoc_req(
&mut ctx,
fidl_ieee80211::ReasonCode::LeavingNetworkDisassoc.into_primitive(),
)
.await
.expect("expected OK");
assert_variant!(r_sta.state.as_ref(), State::Authenticated);
assert_eq!(fake_device_state.lock().wlan_queue.len(), 1);
#[rustfmt::skip]
assert_eq!(&fake_device_state.lock().wlan_queue[0].0[..], &[
0b10100000, 0, 0, 0, 1, 1, 1, 1, 1, 1, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 0x10, 0, 8, 0, ][..]);
}
#[test]
fn handle_mlme_set_controlled_port_req() {
let mut r_sta = make_remote_client();
r_sta.state = StateMachine::new(State::Associated {
aid: 1,
eapol_controlled_port: Some(fidl_mlme::ControlledPortState::Closed),
active_timeout_event_id: None,
ps_state: PowerSaveState::Awake,
});
r_sta
.handle_mlme_set_controlled_port_req(fidl_mlme::ControlledPortState::Open)
.expect("expected OK");
assert_variant!(
r_sta.state.as_ref(),
State::Associated {
eapol_controlled_port: Some(fidl_mlme::ControlledPortState::Open),
..
}
);
}
#[test]
fn handle_mlme_set_controlled_port_req_closed() {
let mut r_sta = make_remote_client();
r_sta.state = StateMachine::new(State::Associated {
aid: 1,
eapol_controlled_port: Some(fidl_mlme::ControlledPortState::Open),
active_timeout_event_id: None,
ps_state: PowerSaveState::Awake,
});
r_sta
.handle_mlme_set_controlled_port_req(fidl_mlme::ControlledPortState::Closed)
.expect("expected OK");
assert_variant!(
r_sta.state.as_ref(),
State::Associated {
eapol_controlled_port: Some(fidl_mlme::ControlledPortState::Closed),
..
}
);
}
#[test]
fn handle_mlme_set_controlled_port_req_no_rsn() {
let mut r_sta = make_remote_client();
r_sta.state = StateMachine::new(State::Associated {
aid: 1,
eapol_controlled_port: None,
active_timeout_event_id: None,
ps_state: PowerSaveState::Awake,
});
assert_eq!(
zx::Status::from(
r_sta
.handle_mlme_set_controlled_port_req(fidl_mlme::ControlledPortState::Open)
.expect_err("expected err")
),
zx::Status::BAD_STATE
);
assert_variant!(
r_sta.state.as_ref(),
State::Associated { eapol_controlled_port: None, .. }
);
}
#[test]
fn handle_mlme_set_controlled_port_req_wrong_state() {
let mut r_sta = make_remote_client();
r_sta.state = StateMachine::new(State::Authenticating);
assert_eq!(
zx::Status::from(
r_sta
.handle_mlme_set_controlled_port_req(fidl_mlme::ControlledPortState::Open)
.expect_err("expected err")
),
zx::Status::BAD_STATE
);
}
#[fuchsia::test(allow_stalls = false)]
async fn handle_mlme_eapol_req() {
let (fake_device, fake_device_state) = FakeDevice::new().await;
let mut r_sta = make_remote_client();
let (mut ctx, _) = make_context(fake_device);
r_sta.handle_mlme_eapol_req(&mut ctx, *CLIENT_ADDR2, &[1, 2, 3][..]).expect("expected OK");
assert_eq!(fake_device_state.lock().wlan_queue.len(), 1);
#[rustfmt::skip]
assert_eq!(&fake_device_state.lock().wlan_queue[0].0[..], &[
0b00001000, 0b00000010, 0, 0, 1, 1, 1, 1, 1, 1, 2, 2, 2, 2, 2, 2, 3, 3, 3, 3, 3, 3, 0x10, 0, 0xAA, 0xAA, 0x03, 0, 0, 0, 0x88, 0x8E, 1, 2, 3,
][..]);
}
#[fuchsia::test(allow_stalls = false)]
async fn handle_disassoc_frame() {
let (fake_device, fake_device_state) = FakeDevice::new().await;
let mut r_sta = make_remote_client();
let (mut ctx, _) = make_context(fake_device);
r_sta
.handle_disassoc_frame(
&mut ctx,
ReasonCode(fidl_ieee80211::ReasonCode::LeavingNetworkDisassoc.into_primitive()),
)
.await
.expect("expected OK");
let msg = fake_device_state
.lock()
.next_mlme_msg::<fidl_mlme::DisassociateIndication>()
.expect("expected MLME message");
assert_eq!(
msg,
fidl_mlme::DisassociateIndication {
peer_sta_address: CLIENT_ADDR.to_array(),
reason_code: fidl_ieee80211::ReasonCode::LeavingNetworkDisassoc,
locally_initiated: false,
},
);
assert_variant!(r_sta.state.as_ref(), State::Authenticated);
}
#[test_case(State::Authenticating; "in authenticating state")]
#[test_case(State::Authenticated; "in authenticated state")]
#[test_case(State::Associated {
aid: 1,
eapol_controlled_port: None,
active_timeout_event_id: None,
ps_state: PowerSaveState::Awake,
}; "in associated state")]
#[fuchsia::test(allow_stalls = false)]
async fn handle_assoc_req_frame(init_state: State) {
let (fake_device, fake_device_state) = FakeDevice::new().await;
let mut r_sta = make_remote_client();
r_sta.state = StateMachine::new(init_state);
let (mut ctx, _) = make_context(fake_device);
r_sta
.handle_assoc_req_frame(
&mut ctx,
CapabilityInfo(0).with_short_preamble(true),
1,
Some(Ssid::try_from("coolnet").unwrap()),
vec![1, 2, 3, 4, 5, 6, 7, 8, 9, 10].iter().map(|r| ie::SupportedRate(*r)).collect(),
None,
)
.expect("expected OK");
let msg = fake_device_state
.lock()
.next_mlme_msg::<fidl_mlme::AssociateIndication>()
.expect("expected MLME message");
assert_eq!(
msg,
fidl_mlme::AssociateIndication {
peer_sta_address: CLIENT_ADDR.to_array(),
listen_interval: 1,
ssid: Some(Ssid::try_from("coolnet").unwrap().into()),
capability_info: CapabilityInfo(0).with_short_preamble(true).raw(),
rates: vec![1, 2, 3, 4, 5, 6, 7, 8, 9, 10],
rsne: None,
},
);
}
#[test_case(State::Authenticating; "in authenticating state")]
#[test_case(State::Authenticated; "in authenticated state")]
#[test_case(State::Associated {
aid: 1,
eapol_controlled_port: None,
active_timeout_event_id: None,
ps_state: PowerSaveState::Awake,
}; "in associated state")]
#[fuchsia::test(allow_stalls = false)]
async fn handle_auth_frame(init_state: State) {
let (fake_device, fake_device_state) = FakeDevice::new().await;
let mut r_sta = make_remote_client();
r_sta.state = StateMachine::new(init_state);
let (mut ctx, _) = make_context(fake_device);
r_sta
.handle_auth_frame(&mut ctx, AuthAlgorithmNumber::SHARED_KEY)
.await
.expect("expected OK");
let msg = fake_device_state
.lock()
.next_mlme_msg::<fidl_mlme::AuthenticateIndication>()
.expect("expected MLME message");
assert_eq!(
msg,
fidl_mlme::AuthenticateIndication {
peer_sta_address: CLIENT_ADDR.to_array(),
auth_type: fidl_mlme::AuthenticationTypes::SharedKey,
},
);
}
#[fuchsia::test(allow_stalls = false)]
async fn handle_auth_frame_unknown_algorithm() {
let (fake_device, fake_device_state) = FakeDevice::new().await;
let mut r_sta = make_remote_client();
let (mut ctx, _) = make_context(fake_device);
r_sta.handle_auth_frame(&mut ctx, AuthAlgorithmNumber(0xffff)).await.expect("expected OK");
assert_eq!(fake_device_state.lock().wlan_queue.len(), 1);
#[rustfmt::skip]
assert_eq!(&fake_device_state.lock().wlan_queue[0].0[..], &[
0b10110000, 0, 0, 0, 1, 1, 1, 1, 1, 1, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 0x10, 0, 0xff, 0xff, 2, 0, 13, 0, ][..]);
assert_variant!(r_sta.state.as_ref(), State::Deauthenticated);
}
#[test_case(false; "from idle state")]
#[test_case(true; "while already authenticated")]
#[fuchsia::test(allow_stalls = false)]
async fn handle_deauth_frame(already_authenticated: bool) {
let (fake_device, fake_device_state) = FakeDevice::new().await;
let mut r_sta = make_remote_client();
if already_authenticated {
r_sta.state = StateMachine::new(State::Authenticated);
}
let (mut ctx, _) = make_context(fake_device);
r_sta
.handle_deauth_frame(
&mut ctx,
ReasonCode(fidl_ieee80211::ReasonCode::LeavingNetworkDeauth.into_primitive()),
)
.await
.expect("expected OK");
let msg = fake_device_state
.lock()
.next_mlme_msg::<fidl_mlme::DeauthenticateIndication>()
.expect("expected MLME message");
assert_eq!(
msg,
fidl_mlme::DeauthenticateIndication {
peer_sta_address: CLIENT_ADDR.to_array(),
reason_code: fidl_ieee80211::ReasonCode::LeavingNetworkDeauth,
locally_initiated: false,
}
);
assert_variant!(r_sta.state.as_ref(), State::Deauthenticated);
}
#[test]
fn handle_action_frame() {
}
#[fuchsia::test(allow_stalls = false)]
async fn handle_ps_poll() {
let (fake_device, fake_device_state) = FakeDevice::new().await;
let (mut ctx, _) = make_context(fake_device);
let mut r_sta = make_remote_client();
r_sta.state = StateMachine::new(State::Associated {
aid: 1,
eapol_controlled_port: None,
active_timeout_event_id: None,
ps_state: PowerSaveState::Awake,
});
r_sta.set_power_state(&mut ctx, mac::PowerState::DOZE).expect("expected doze OK");
r_sta
.handle_eth_frame(
&mut ctx,
*CLIENT_ADDR2,
*CLIENT_ADDR,
0x1234,
&[1, 2, 3, 4, 5][..],
0.into(),
)
.expect("expected OK");
r_sta
.handle_eth_frame(
&mut ctx,
*CLIENT_ADDR2,
*CLIENT_ADDR,
0x1234,
&[6, 7, 8, 9, 0][..],
0.into(),
)
.expect("expected OK");
assert_eq!(fake_device_state.lock().wlan_queue.len(), 0);
r_sta.handle_ps_poll(&mut ctx, 1).expect("expected handle_ps_poll OK");
assert_eq!(fake_device_state.lock().wlan_queue.len(), 1);
assert_eq!(
&fake_device_state.lock().wlan_queue[0].0[..],
&[
0b00001000, 0b00100010, 0, 0, 3, 3, 3, 3, 3, 3, 2, 2, 2, 2, 2, 2, 1, 1, 1, 1, 1, 1, 0x10, 0, 0xAA, 0xAA, 0x03, 0, 0, 0, 0x12, 0x34, 1, 2, 3, 4, 5,
][..]
);
r_sta.handle_ps_poll(&mut ctx, 1).expect("expected handle_ps_poll OK");
assert_eq!(fake_device_state.lock().wlan_queue.len(), 2);
assert_eq!(
&fake_device_state.lock().wlan_queue[1].0[..],
&[
0b00001000, 0b00000010, 0, 0, 3, 3, 3, 3, 3, 3, 2, 2, 2, 2, 2, 2, 1, 1, 1, 1, 1, 1, 0x20, 0, 0xAA, 0xAA, 0x03, 0, 0, 0, 0x12, 0x34, 6, 7, 8, 9, 0,
][..]
);
r_sta.handle_ps_poll(&mut ctx, 1).expect("expected handle_ps_poll OK");
assert_eq!(fake_device_state.lock().wlan_queue.len(), 2);
}
#[fuchsia::test(allow_stalls = false)]
async fn handle_ps_poll_not_buffered() {
let (fake_device, _) = FakeDevice::new().await;
let (mut ctx, _) = make_context(fake_device);
let mut r_sta = make_remote_client();
r_sta.state = StateMachine::new(State::Associated {
aid: 1,
eapol_controlled_port: None,
active_timeout_event_id: None,
ps_state: PowerSaveState::Awake,
});
r_sta.set_power_state(&mut ctx, mac::PowerState::DOZE).expect("expected doze OK");
r_sta.handle_ps_poll(&mut ctx, 1).expect("expected handle_ps_poll OK");
}
#[fuchsia::test(allow_stalls = false)]
async fn handle_ps_poll_wrong_aid() {
let (fake_device, _) = FakeDevice::new().await;
let (mut ctx, _) = make_context(fake_device);
let mut r_sta = make_remote_client();
r_sta.state = StateMachine::new(State::Associated {
aid: 1,
eapol_controlled_port: None,
active_timeout_event_id: None,
ps_state: PowerSaveState::Awake,
});
r_sta.set_power_state(&mut ctx, mac::PowerState::DOZE).expect("expected doze OK");
assert_variant!(
r_sta.handle_ps_poll(&mut ctx, 2).expect_err("expected handle_ps_poll error"),
ClientRejection::NotPermitted
);
}
#[fuchsia::test(allow_stalls = false)]
async fn handle_ps_poll_not_dozing() {
let (fake_device, _) = FakeDevice::new().await;
let (mut ctx, _) = make_context(fake_device);
let mut r_sta = make_remote_client();
r_sta.state = StateMachine::new(State::Associated {
aid: 1,
eapol_controlled_port: None,
active_timeout_event_id: None,
ps_state: PowerSaveState::Awake,
});
assert_variant!(
r_sta.handle_ps_poll(&mut ctx, 1).expect_err("expected handle_ps_poll error"),
ClientRejection::NotPermitted
);
}
#[fuchsia::test(allow_stalls = false)]
async fn handle_eapol_llc_frame() {
let (fake_device, fake_device_state) = FakeDevice::new().await;
let mut r_sta = make_remote_client();
let (mut ctx, _) = make_context(fake_device);
r_sta.state = StateMachine::new(State::Associated {
aid: 1,
eapol_controlled_port: None,
active_timeout_event_id: None,
ps_state: PowerSaveState::Awake,
});
r_sta
.handle_eapol_llc_frame(&mut ctx, *CLIENT_ADDR2, *CLIENT_ADDR, &[1, 2, 3, 4, 5][..])
.expect("expected OK");
let msg = fake_device_state
.lock()
.next_mlme_msg::<fidl_mlme::EapolIndication>()
.expect("expected MLME message");
assert_eq!(
msg,
fidl_mlme::EapolIndication {
dst_addr: CLIENT_ADDR2.to_array(),
src_addr: CLIENT_ADDR.to_array(),
data: vec![1, 2, 3, 4, 5],
},
);
}
#[fuchsia::test(allow_stalls = false)]
async fn handle_llc_frame() {
let (fake_device, fake_device_state) = FakeDevice::new().await;
let mut r_sta = make_remote_client();
let (mut ctx, _) = make_context(fake_device);
r_sta.state = StateMachine::new(State::Associated {
aid: 1,
eapol_controlled_port: None,
active_timeout_event_id: None,
ps_state: PowerSaveState::Awake,
});
r_sta
.handle_llc_frame(&mut ctx, *CLIENT_ADDR2, *CLIENT_ADDR, 0x1234, &[1, 2, 3, 4, 5][..])
.expect("expected OK");
assert_eq!(fake_device_state.lock().eth_queue.len(), 1);
#[rustfmt::skip]
assert_eq!(&fake_device_state.lock().eth_queue[0][..], &[
3, 3, 3, 3, 3, 3, 1, 1, 1, 1, 1, 1, 0x12, 0x34, 1, 2, 3, 4, 5,
][..]);
}
#[fuchsia::test(allow_stalls = false)]
async fn handle_eth_frame_no_eapol_controlled_port() {
let (fake_device, fake_device_state) = FakeDevice::new().await;
let mut r_sta = make_remote_client();
let (mut ctx, _) = make_context(fake_device);
r_sta.state = StateMachine::new(State::Associated {
aid: 1,
eapol_controlled_port: None,
active_timeout_event_id: None,
ps_state: PowerSaveState::Awake,
});
r_sta
.handle_eth_frame(
&mut ctx,
*CLIENT_ADDR2,
*CLIENT_ADDR,
0x1234,
&[1, 2, 3, 4, 5][..],
0.into(),
)
.expect("expected OK");
assert_eq!(fake_device_state.lock().wlan_queue.len(), 1);
#[rustfmt::skip]
assert_eq!(&fake_device_state.lock().wlan_queue[0].0[..], &[
0b00001000, 0b00000010, 0, 0, 3, 3, 3, 3, 3, 3, 2, 2, 2, 2, 2, 2, 1, 1, 1, 1, 1, 1, 0x10, 0, 0xAA, 0xAA, 0x03, 0, 0, 0, 0x12, 0x34, 1, 2, 3, 4, 5,
][..]);
}
#[fuchsia::test(allow_stalls = false)]
async fn handle_eth_frame_not_associated() {
let (fake_device, _) = FakeDevice::new().await;
let mut r_sta = make_remote_client();
let (mut ctx, _) = make_context(fake_device);
r_sta.state = StateMachine::new(State::Authenticated);
assert_variant!(
r_sta
.handle_eth_frame(
&mut ctx,
*CLIENT_ADDR2,
*CLIENT_ADDR,
0x1234,
&[1, 2, 3, 4, 5][..],
0.into()
)
.expect_err("expected error"),
ClientRejection::NotAssociated
);
}
#[fuchsia::test(allow_stalls = false)]
async fn handle_eth_frame_eapol_controlled_port_closed() {
let (fake_device, _) = FakeDevice::new().await;
let mut r_sta = make_remote_client();
let (mut ctx, _) = make_context(fake_device);
r_sta.state = StateMachine::new(State::Associated {
aid: 1,
eapol_controlled_port: Some(fidl_mlme::ControlledPortState::Closed),
active_timeout_event_id: None,
ps_state: PowerSaveState::Awake,
});
assert_variant!(
r_sta
.handle_eth_frame(
&mut ctx,
*CLIENT_ADDR2,
*CLIENT_ADDR,
0x1234,
&[1, 2, 3, 4, 5][..],
0.into()
)
.expect_err("expected error"),
ClientRejection::ControlledPortClosed
);
}
#[fuchsia::test(allow_stalls = false)]
async fn handle_eth_frame_eapol_controlled_port_open() {
let (fake_device, fake_device_state) = FakeDevice::new().await;
let mut r_sta = make_remote_client();
let (mut ctx, _) = make_context(fake_device);
r_sta.state = StateMachine::new(State::Associated {
aid: 1,
eapol_controlled_port: Some(fidl_mlme::ControlledPortState::Open),
active_timeout_event_id: None,
ps_state: PowerSaveState::Awake,
});
r_sta
.handle_eth_frame(
&mut ctx,
*CLIENT_ADDR2,
*CLIENT_ADDR,
0x1234,
&[1, 2, 3, 4, 5][..],
0.into(),
)
.expect("expected OK");
assert_eq!(fake_device_state.lock().wlan_queue.len(), 1);
#[rustfmt::skip]
assert_eq!(&fake_device_state.lock().wlan_queue[0].0[..], &[
0b00001000, 0b01000010, 0, 0, 3, 3, 3, 3, 3, 3, 2, 2, 2, 2, 2, 2, 1, 1, 1, 1, 1, 1, 0x10, 0, 0xAA, 0xAA, 0x03, 0, 0, 0, 0x12, 0x34, 1, 2, 3, 4, 5,
][..]);
}
#[fuchsia::test(allow_stalls = false)]
async fn handle_data_frame_not_permitted() {
let (fake_device, fake_device_state) = FakeDevice::new().await;
let mut r_sta = make_remote_client();
r_sta.state = StateMachine::new(State::Authenticating);
let (mut ctx, _) = make_context(fake_device);
assert_variant!(
r_sta
.handle_data_frame(
&mut ctx,
mac::DataFrame {
fixed_fields: mac::FixedDataHdrFields {
frame_ctrl: mac::FrameControl(0b000000010_00001000),
duration: 0,
addr1: *CLIENT_ADDR,
addr2: (*AP_ADDR).into(),
addr3: *CLIENT_ADDR2,
seq_ctrl: mac::SequenceControl(10),
}
.as_bytes_ref(),
addr4: None,
qos_ctrl: None,
ht_ctrl: None,
body: &[
7, 7, 7, 8, 8, 8, 9, 10, 11, 11, 11,
][..],
},
)
.expect_err("expected err"),
ClientRejection::NotPermitted
);
let msg = fake_device_state
.lock()
.next_mlme_msg::<fidl_mlme::DeauthenticateIndication>()
.expect("expected MLME message");
assert_eq!(
msg,
fidl_mlme::DeauthenticateIndication {
peer_sta_address: CLIENT_ADDR.to_array(),
reason_code: fidl_ieee80211::ReasonCode::InvalidClass3Frame,
locally_initiated: true,
},
);
assert_eq!(fake_device_state.lock().wlan_queue.len(), 1);
assert_eq!(
fake_device_state.lock().wlan_queue[0].0,
&[
0b11000000, 0b00000000, 0, 0, 1, 1, 1, 1, 1, 1, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 0x10, 0, 7, 0, ][..]
);
}
#[fuchsia::test(allow_stalls = false)]
async fn handle_data_frame_not_permitted_disassoc() {
let (fake_device, fake_device_state) = FakeDevice::new().await;
let mut r_sta = make_remote_client();
r_sta.state = StateMachine::new(State::Authenticated);
let (mut ctx, _) = make_context(fake_device);
assert_variant!(
r_sta
.handle_data_frame(
&mut ctx,
mac::DataFrame {
fixed_fields: mac::FixedDataHdrFields {
frame_ctrl: mac::FrameControl(0b000000010_00001000),
duration: 0,
addr1: *CLIENT_ADDR,
addr2: (*AP_ADDR).into(),
addr3: *CLIENT_ADDR2,
seq_ctrl: mac::SequenceControl(10),
}
.as_bytes_ref(),
addr4: None,
qos_ctrl: None,
ht_ctrl: None,
body: &[
7, 7, 7, 8, 8, 8, 9, 10, 11, 11, 11,
][..],
},
)
.expect_err("expected err"),
ClientRejection::NotPermitted
);
let msg = fake_device_state
.lock()
.next_mlme_msg::<fidl_mlme::DisassociateIndication>()
.expect("expected MLME message");
assert_eq!(
msg,
fidl_mlme::DisassociateIndication {
peer_sta_address: CLIENT_ADDR.to_array(),
reason_code: fidl_ieee80211::ReasonCode::InvalidClass3Frame,
locally_initiated: true,
},
);
assert_eq!(fake_device_state.lock().wlan_queue.len(), 1);
assert_eq!(
fake_device_state.lock().wlan_queue[0].0,
&[
0b10100000, 0b00000000, 0, 0, 1, 1, 1, 1, 1, 1, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 0x10, 0, 7, 0, ][..]
);
}
#[fuchsia::test(allow_stalls = false)]
async fn handle_data_frame_single_llc() {
let (fake_device, fake_device_state) = FakeDevice::new().await;
let mut r_sta = make_remote_client();
r_sta.state = StateMachine::new(State::Associated {
aid: 1,
eapol_controlled_port: None,
active_timeout_event_id: None,
ps_state: PowerSaveState::Awake,
});
let (mut ctx, _) = make_context(fake_device);
r_sta
.handle_data_frame(
&mut ctx,
mac::DataFrame {
fixed_fields: mac::FixedDataHdrFields {
frame_ctrl: mac::FrameControl(0b000000010_00001000),
duration: 0,
addr1: *CLIENT_ADDR,
addr2: (*AP_ADDR).into(),
addr3: *CLIENT_ADDR2,
seq_ctrl: mac::SequenceControl(10),
}
.as_bytes_ref(),
addr4: None,
qos_ctrl: None,
ht_ctrl: None,
body: &[
7, 7, 7, 8, 8, 8, 9, 10, 11, 11, 11,
][..],
},
)
.expect("expected OK");
assert_eq!(fake_device_state.lock().eth_queue.len(), 1);
assert_ne!(
match r_sta.state.as_ref() {
State::Associated { active_timeout_event_id, .. } => *active_timeout_event_id,
_ => panic!("expected Associated"),
},
None
)
}
#[fuchsia::test(allow_stalls = false)]
async fn handle_data_frame_amsdu() {
let (fake_device, fake_device_state) = FakeDevice::new().await;
let mut r_sta = make_remote_client();
r_sta.state = StateMachine::new(State::Associated {
aid: 1,
eapol_controlled_port: None,
active_timeout_event_id: None,
ps_state: PowerSaveState::Awake,
});
let (mut ctx, _) = make_context(fake_device);
let mut amsdu_data_frame_body = vec![];
amsdu_data_frame_body.extend(&[
0x78, 0x8a, 0x20, 0x0d, 0x67, 0x03, 0xb4, 0xf7, 0xa1, 0xbe, 0xb9, 0xab, 0x00, 0x74, ]);
amsdu_data_frame_body.extend(MSDU_1_LLC_HDR);
amsdu_data_frame_body.extend(MSDU_1_PAYLOAD);
amsdu_data_frame_body.extend(&[
0x00, 0x00, 0x78, 0x8a, 0x20, 0x0d, 0x67, 0x04, 0xb4, 0xf7, 0xa1, 0xbe, 0xb9, 0xac, 0x00, 0x66, ]);
amsdu_data_frame_body.extend(MSDU_2_LLC_HDR);
amsdu_data_frame_body.extend(MSDU_2_PAYLOAD);
r_sta
.handle_data_frame(
&mut ctx,
mac::DataFrame {
fixed_fields: mac::FixedDataHdrFields {
frame_ctrl: mac::FrameControl(0b000000010_00001000),
duration: 0,
addr1: *CLIENT_ADDR,
addr2: (*AP_ADDR).into(),
addr3: *CLIENT_ADDR2,
seq_ctrl: mac::SequenceControl(10),
}
.as_bytes_ref(),
addr4: None,
qos_ctrl: Some(
Unalign::new(mac::QosControl(0).with_amsdu_present(true)).as_bytes_ref(),
),
ht_ctrl: None,
body: &amsdu_data_frame_body[..],
},
)
.expect("expected OK");
assert_eq!(fake_device_state.lock().eth_queue.len(), 2);
assert_ne!(
match r_sta.state.as_ref() {
State::Associated { active_timeout_event_id, .. } => *active_timeout_event_id,
_ => panic!("expected Associated"),
},
None
)
}
#[fuchsia::test(allow_stalls = false)]
async fn handle_mgmt_frame() {
let (fake_device, _) = FakeDevice::new().await;
let mut r_sta = make_remote_client();
r_sta.state = StateMachine::new(State::Authenticating);
let (mut ctx, _) = make_context(fake_device);
r_sta
.handle_mgmt_frame(
&mut ctx,
mac::CapabilityInfo(0),
None,
mac::MgmtFrame {
mgmt_hdr: mac::MgmtHdr {
frame_ctrl: mac::FrameControl(0b00000000_10110000), duration: 0,
addr1: [1; 6].into(),
addr2: [2; 6].into(),
addr3: [3; 6].into(),
seq_ctrl: mac::SequenceControl(10),
}
.as_bytes_ref(),
ht_ctrl: None,
body: &[
0, 0, 1, 0, 0, 0, ][..],
},
)
.await
.expect("expected OK");
}
#[test_case(Ssid::try_from("coolnet").unwrap(), true; "with ssid and rsne")]
#[test_case(Ssid::try_from("").unwrap(), true; "with empty ssid")]
#[test_case(Ssid::try_from("coolnet").unwrap(), false; "without rsne")]
#[fuchsia::test(allow_stalls = false)]
async fn handle_mgmt_frame_assoc_req(ssid: Ssid, has_rsne: bool) {
let (fake_device, fake_device_state) = FakeDevice::new().await;
let mut r_sta = make_remote_client();
r_sta.state = StateMachine::new(State::Authenticated);
let (mut ctx, _) = make_context(fake_device);
let mut assoc_frame_body = vec![
0, 0, 10, 0, 1, 8, 1, 2, 3, 4, 5, 6, 7, 8, 50, 2, 9, 10, ];
if has_rsne {
assoc_frame_body.extend(&[48, 2, 77, 88][..]); }
r_sta
.handle_mgmt_frame(
&mut ctx,
mac::CapabilityInfo(0),
Some(ssid.clone()),
mac::MgmtFrame {
mgmt_hdr: mac::MgmtHdr {
frame_ctrl: mac::FrameControl(0b00000000_00000000), duration: 0,
addr1: [1; 6].into(),
addr2: [2; 6].into(),
addr3: [3; 6].into(),
seq_ctrl: mac::SequenceControl(10),
}
.as_bytes_ref(),
ht_ctrl: None,
body: &assoc_frame_body[..],
},
)
.await
.expect("expected OK");
let msg = fake_device_state
.lock()
.next_mlme_msg::<fidl_mlme::AssociateIndication>()
.expect("expected MLME message");
let expected_rsne = if has_rsne { Some(vec![48, 2, 77, 88]) } else { None };
assert_eq!(
msg,
fidl_mlme::AssociateIndication {
peer_sta_address: CLIENT_ADDR.to_array(),
listen_interval: 10,
ssid: Some(ssid.into()),
capability_info: CapabilityInfo(0).raw(),
rates: vec![1, 2, 3, 4, 5, 6, 7, 8, 9, 10],
rsne: expected_rsne,
},
);
}
#[test_case(vec![1, 0],
vec![50, 2, 9, 10],
vec![9, 10] ; "when no supported rates")]
#[test_case(vec![1, 8, 1, 2, 3, 4, 5, 6, 7, 8],
vec![50, 0],
vec![1, 2, 3, 4, 5, 6, 7, 8] ; "when no extended supported rates")]
#[test_case(vec![1, 0],
vec![50, 0],
vec![] ; "when no rates")]
#[test_case(vec![1, 9, 1, 2, 3, 4, 5, 6, 7, 8, 9],
vec![50, 9, 10],
vec![1, 2, 3, 4, 5, 6, 7, 8, 9] ; "when too many supported rates")]
#[fuchsia::test(allow_stalls = false)]
async fn assoc_req_with_bad_rates_still_passed_to_sme(
supported_rates_ie: Vec<u8>,
extended_supported_rates_ie: Vec<u8>,
expected_rates: Vec<u8>,
) {
let (fake_device, fake_device_state) = FakeDevice::new().await;
let mut r_sta = make_remote_client();
r_sta.state = StateMachine::new(State::Authenticated);
let (mut ctx, _) = make_context(fake_device);
let mut ies = vec![
0, 0, 10, 0, ];
ies.extend(supported_rates_ie);
ies.extend(extended_supported_rates_ie);
r_sta
.handle_mgmt_frame(
&mut ctx,
mac::CapabilityInfo(0),
Some(Ssid::try_from("coolnet").unwrap()),
mac::MgmtFrame {
mgmt_hdr: mac::MgmtHdr {
frame_ctrl: mac::FrameControl(0b00000000_00000000), duration: 0,
addr1: [1; 6].into(),
addr2: [2; 6].into(),
addr3: [3; 6].into(),
seq_ctrl: mac::SequenceControl(10),
}
.as_bytes_ref(),
ht_ctrl: None,
body: &ies[..],
},
)
.await
.expect("parsing should not fail");
let msg = fake_device_state
.lock()
.next_mlme_msg::<fidl_mlme::AssociateIndication>()
.expect("expected MLME message");
assert_eq!(
msg,
fidl_mlme::AssociateIndication {
peer_sta_address: CLIENT_ADDR.to_array(),
listen_interval: 10,
ssid: Some(Ssid::try_from("coolnet").unwrap().into()),
capability_info: CapabilityInfo(0).raw(),
rates: expected_rates,
rsne: None,
},
);
}
#[fuchsia::test(allow_stalls = false)]
async fn handle_mgmt_frame_not_permitted() {
let (fake_device, fake_device_state) = FakeDevice::new().await;
let mut r_sta = make_remote_client();
r_sta.state = StateMachine::new(State::Authenticating);
let (mut ctx, _) = make_context(fake_device);
assert_variant!(
r_sta
.handle_mgmt_frame(
&mut ctx,
mac::CapabilityInfo(0),
None,
mac::MgmtFrame {
mgmt_hdr: mac::MgmtHdr {
frame_ctrl: mac::FrameControl(0b00000000_00000000), duration: 0,
addr1: [1; 6].into(),
addr2: [2; 6].into(),
addr3: [3; 6].into(),
seq_ctrl: mac::SequenceControl(10),
}
.as_bytes_ref(),
ht_ctrl: None,
body: &[
0, 0, 10, 0, ][..],
},
)
.await
.expect_err("expected error"),
ClientRejection::NotPermitted
);
let msg = fake_device_state
.lock()
.next_mlme_msg::<fidl_mlme::DeauthenticateIndication>()
.expect("expected MLME message");
assert_eq!(
msg,
fidl_mlme::DeauthenticateIndication {
peer_sta_address: CLIENT_ADDR.to_array(),
reason_code: fidl_ieee80211::ReasonCode::InvalidClass2Frame,
locally_initiated: true,
},
);
assert_eq!(fake_device_state.lock().wlan_queue.len(), 1);
assert_eq!(
fake_device_state.lock().wlan_queue[0].0,
&[
0b11000000, 0b00000000, 0, 0, 1, 1, 1, 1, 1, 1, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 0x10, 0, 6, 0, ][..]
);
}
#[fuchsia::test(allow_stalls = false)]
async fn handle_mgmt_frame_not_handled() {
let (fake_device, _) = FakeDevice::new().await;
let mut r_sta = make_remote_client();
r_sta.state = StateMachine::new(State::Associated {
aid: 1,
eapol_controlled_port: None,
active_timeout_event_id: None,
ps_state: PowerSaveState::Awake,
});
let (mut ctx, _) = make_context(fake_device);
assert_variant!(
r_sta
.handle_mgmt_frame(
&mut ctx,
mac::CapabilityInfo(0),
None,
mac::MgmtFrame {
mgmt_hdr: mac::MgmtHdr {
frame_ctrl: mac::FrameControl(0b00000000_00010000), duration: 0,
addr1: [1; 6].into(),
addr2: [2; 6].into(),
addr3: [3; 6].into(),
seq_ctrl: mac::SequenceControl(10),
}
.as_bytes_ref(),
ht_ctrl: None,
body: &[
0, 0, 0, 0, 1, 0, ][..],
},
)
.await
.expect_err("expected error"),
ClientRejection::Unsupported
);
}
#[fuchsia::test(allow_stalls = false)]
async fn handle_mgmt_frame_resets_active_timer() {
let (fake_device, _) = FakeDevice::new().await;
let mut r_sta = make_remote_client();
r_sta.state = StateMachine::new(State::Associated {
aid: 1,
eapol_controlled_port: None,
active_timeout_event_id: None,
ps_state: PowerSaveState::Awake,
});
let (mut ctx, _) = make_context(fake_device);
r_sta
.handle_mgmt_frame(
&mut ctx,
mac::CapabilityInfo(0),
None,
mac::MgmtFrame {
mgmt_hdr: mac::MgmtHdr {
frame_ctrl: mac::FrameControl(0b00000000_00000000), duration: 0,
addr1: [1; 6].into(),
addr2: [2; 6].into(),
addr3: [3; 6].into(),
seq_ctrl: mac::SequenceControl(10),
}
.as_bytes_ref(),
ht_ctrl: None,
body: &[
0, 0, 10, 0, ][..],
},
)
.await
.expect("expected OK");
assert_ne!(
match r_sta.state.as_ref() {
State::Associated { active_timeout_event_id, .. } => *active_timeout_event_id,
_ => panic!("expected Associated"),
},
None
)
}
#[fuchsia::test(allow_stalls = false)]
async fn handle_bss_idle_timeout() {
let (fake_device, fake_device_state) = FakeDevice::new().await;
let (mut ctx, _) = make_context(fake_device);
let mut r_sta = make_remote_client();
let event_id = r_sta.schedule_bss_idle_timeout(&mut ctx);
r_sta.state = StateMachine::new(State::Associated {
aid: 1,
eapol_controlled_port: None,
active_timeout_event_id: Some(event_id),
ps_state: PowerSaveState::Awake,
});
r_sta.handle_bss_idle_timeout(&mut ctx, event_id).await.expect("expected OK");
assert_variant!(r_sta.state.as_ref(), State::Authenticated);
assert_eq!(fake_device_state.lock().wlan_queue.len(), 1);
#[rustfmt::skip]
assert_eq!(&fake_device_state.lock().wlan_queue[0].0[..], &[
0b10100000, 0, 0, 0, 1, 1, 1, 1, 1, 1, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 0x10, 0, 4, 0, ][..]);
let msg = fake_device_state
.lock()
.next_mlme_msg::<fidl_mlme::DisassociateIndication>()
.expect("expected MLME message");
assert_eq!(
msg,
fidl_mlme::DisassociateIndication {
peer_sta_address: CLIENT_ADDR.to_array(),
reason_code: fidl_ieee80211::ReasonCode::ReasonInactivity,
locally_initiated: true,
},
);
}
#[fuchsia::test(allow_stalls = false)]
async fn doze_then_wake() {
let (fake_device, fake_device_state) = FakeDevice::new().await;
let (mut ctx, _) = make_context(fake_device);
let mut r_sta = make_remote_client();
r_sta.state = StateMachine::new(State::Associated {
aid: 1,
eapol_controlled_port: None,
active_timeout_event_id: None,
ps_state: PowerSaveState::Awake,
});
r_sta.set_power_state(&mut ctx, mac::PowerState::DOZE).expect("expected doze OK");
r_sta
.handle_eth_frame(
&mut ctx,
*CLIENT_ADDR2,
*CLIENT_ADDR,
0x1234,
&[1, 2, 3, 4, 5][..],
0.into(),
)
.expect("expected OK");
r_sta
.handle_eth_frame(
&mut ctx,
*CLIENT_ADDR2,
*CLIENT_ADDR,
0x1234,
&[6, 7, 8, 9, 0][..],
0.into(),
)
.expect("expected OK");
assert!(r_sta.has_buffered_frames());
assert_eq!(fake_device_state.lock().wlan_queue.len(), 0);
r_sta.set_power_state(&mut ctx, mac::PowerState::AWAKE).expect("expected wake OK");
assert!(!r_sta.has_buffered_frames());
assert_eq!(fake_device_state.lock().wlan_queue.len(), 2);
assert_eq!(
&fake_device_state.lock().wlan_queue[0].0[..],
&[
0b00001000, 0b00100010, 0, 0, 3, 3, 3, 3, 3, 3, 2, 2, 2, 2, 2, 2, 1, 1, 1, 1, 1, 1, 0x10, 0, 0xAA, 0xAA, 0x03, 0, 0, 0, 0x12, 0x34, 1, 2, 3, 4, 5,
][..]
);
assert_eq!(
&fake_device_state.lock().wlan_queue[1].0[..],
&[
0b00001000, 0b00000010, 0, 0, 3, 3, 3, 3, 3, 3, 2, 2, 2, 2, 2, 2, 1, 1, 1, 1, 1, 1, 0x20, 0, 0xAA, 0xAA, 0x03, 0, 0, 0, 0x12, 0x34, 6, 7, 8, 9, 0,
][..]
);
}
#[fuchsia::test(allow_stalls = false)]
async fn doze_then_doze() {
let (fake_device, _) = FakeDevice::new().await;
let (mut ctx, _) = make_context(fake_device);
let mut r_sta = make_remote_client();
r_sta.state = StateMachine::new(State::Associated {
aid: 1,
eapol_controlled_port: None,
active_timeout_event_id: None,
ps_state: PowerSaveState::Awake,
});
r_sta.set_power_state(&mut ctx, mac::PowerState::DOZE).expect("expected doze OK");
r_sta.set_power_state(&mut ctx, mac::PowerState::DOZE).expect("expected doze OK");
}
#[fuchsia::test(allow_stalls = false)]
async fn wake_then_wake() {
let (fake_device, _) = FakeDevice::new().await;
let (mut ctx, _) = make_context(fake_device);
let mut r_sta = make_remote_client();
r_sta.state = StateMachine::new(State::Associated {
aid: 1,
eapol_controlled_port: None,
active_timeout_event_id: None,
ps_state: PowerSaveState::Awake,
});
r_sta.set_power_state(&mut ctx, mac::PowerState::AWAKE).expect("expected wake OK");
r_sta.set_power_state(&mut ctx, mac::PowerState::AWAKE).expect("expected wake OK");
}
#[fuchsia::test(allow_stalls = false)]
async fn doze_not_associated() {
let (fake_device, _) = FakeDevice::new().await;
let (mut ctx, _) = make_context(fake_device);
let mut r_sta = make_remote_client();
r_sta.state = StateMachine::new(State::Authenticating);
assert_variant!(
r_sta
.set_power_state(&mut ctx, mac::PowerState::DOZE)
.expect_err("expected doze error"),
ClientRejection::NotAssociated
);
}
#[fuchsia::test(allow_stalls = false)]
async fn wake_not_associated() {
let (fake_device, _) = FakeDevice::new().await;
let (mut ctx, _) = make_context(fake_device);
let mut r_sta = make_remote_client();
r_sta.state = StateMachine::new(State::Authenticating);
r_sta.set_power_state(&mut ctx, mac::PowerState::AWAKE).expect("expected wake OK");
}
}