wlan_rsn/key/exchange/handshake/group_key/
mod.rsuse crate::key::exchange::handshake::group_key::supplicant::Supplicant;
use crate::key::exchange::{self};
use crate::rsna::{Dot11VerifiedKeyFrame, NegotiatedProtection, Role, UpdateSink};
use crate::{rsn_ensure, Error};
use bytes::Bytes;
use zerocopy::SplitByteSlice;
mod supplicant;
#[derive(Debug, PartialEq)]
enum RoleHandler {
Supplicant(Supplicant),
}
pub struct GroupKeyHandshakeFrame<B: SplitByteSlice> {
frame: Dot11VerifiedKeyFrame<B>,
}
impl<B: SplitByteSlice> GroupKeyHandshakeFrame<B> {
#[allow(clippy::result_large_err, reason = "mass allow for https://fxbug.dev/381896734")]
pub fn from_verified(frame: Dot11VerifiedKeyFrame<B>, role: Role) -> Result<Self, Error> {
let raw_frame = frame.unsafe_get_raw();
let sender = match role {
Role::Supplicant => Role::Authenticator,
Role::Authenticator => Role::Supplicant,
};
let key_info = raw_frame.key_frame_fields.key_info();
rsn_ensure!(
key_info.key_type() == eapol::KeyType::GROUP_SMK,
"only group key messages are allowed in Group Key Handshake"
);
rsn_ensure!(
!key_info.install(),
"installbit must not be set in Group Key Handshake messages"
);
rsn_ensure!(key_info.key_mic(), "MIC bit must be set in Group Key Handshake messages");
rsn_ensure!(key_info.secure(), "secure bit must be set in Group Key Handshake messages");
rsn_ensure!(!key_info.error(), "error bit must not be set in Group Key Handshake messages");
rsn_ensure!(
!key_info.request(),
"request bit must not be set in Group Key Handshake messages"
);
match &sender {
Role::Authenticator => {
rsn_ensure!(
key_info.key_ack(),
"ACK bit must be set in 1st message of Group Key Handshake"
);
if raw_frame.key_frame_fields.descriptor_type != eapol::KeyDescriptor::LEGACY_WPA1 {
rsn_ensure!(
key_info.encrypted_key_data(),
"encrypted data bit must be set in 1st message of Group Key Handshake"
);
}
}
Role::Supplicant => {
rsn_ensure!(
!key_info.key_ack(),
"ACK bit must not be set in 2nd message of Group Key Handshake"
);
rsn_ensure!(
!key_info.encrypted_key_data(),
"encrypted data bit must not be set in 2nd message of Group Key Handshake"
);
rsn_ensure!(
is_zero(&raw_frame.key_frame_fields.key_iv[..]),
"IV must be zero in 2nd message of Group Key Handshake"
);
rsn_ensure!(
raw_frame.key_frame_fields.key_rsc.to_native() == 0,
"RSC must be zero in 2nd message of Group Key Handshake"
);
}
};
Ok(GroupKeyHandshakeFrame { frame })
}
pub fn get(self) -> Dot11VerifiedKeyFrame<B> {
self.frame
}
}
#[derive(Debug, Clone, PartialEq)]
pub struct Config {
pub role: Role,
pub protection: NegotiatedProtection,
}
#[derive(Debug, PartialEq)]
pub struct GroupKey(RoleHandler);
impl GroupKey {
#[allow(clippy::result_large_err, reason = "mass allow for https://fxbug.dev/381896734")]
pub fn new(cfg: Config, kck: &[u8], kek: &[u8]) -> Result<GroupKey, Error> {
let handler = match &cfg.role {
Role::Supplicant => RoleHandler::Supplicant(Supplicant {
cfg,
kck: Bytes::copy_from_slice(kck),
kek: Bytes::copy_from_slice(kek),
}),
_ => {
return Err(Error::GenericError(
"Authenticator not yet support in Group-Key Handshake".to_string(),
))
}
};
Ok(GroupKey(handler))
}
pub fn destroy(self) -> exchange::Config {
match self.0 {
RoleHandler::Supplicant(s) => exchange::Config::GroupKeyHandshake(s.destroy()),
}
}
#[allow(clippy::result_large_err, reason = "mass allow for https://fxbug.dev/381896734")]
pub fn on_eapol_key_frame<B: SplitByteSlice>(
&mut self,
update_sink: &mut UpdateSink,
frame: Dot11VerifiedKeyFrame<B>,
) -> Result<(), Error> {
match &mut self.0 {
RoleHandler::Supplicant(s) => {
let frame = GroupKeyHandshakeFrame::from_verified(frame, Role::Supplicant)?;
s.on_eapol_key_frame(update_sink, frame)
}
}
}
}
fn is_zero(slice: &[u8]) -> bool {
slice.iter().all(|&x| x == 0)
}
#[cfg(test)]
mod tests {
use super::*;
use crate::rsna::test_util;
use wlan_common::ie::rsn::fake_wpa2_s_rsne;
fn verify_group_key_frame(key_frame: eapol::KeyFrameBuf, role: Role) {
let protection =
NegotiatedProtection::from_rsne(&fake_wpa2_s_rsne()).expect("error getting RNSE");
let parsed_frame = eapol::KeyFrameRx::parse(test_util::mic_len(), &key_frame[..])
.expect("failed to parse group key frame");
let frame = Dot11VerifiedKeyFrame::from_frame(parsed_frame, &role, &protection, 0)
.expect("couldn't verify frame");
GroupKeyHandshakeFrame::from_verified(frame, role).expect("error verifying group_frame");
}
fn fake_key_frame() -> eapol::KeyFrameTx {
let mut key_frame_fields = eapol::KeyFrameFields::default();
key_frame_fields.descriptor_type = eapol::KeyDescriptor::IEEE802DOT11;
key_frame_fields.set_key_info(eapol::KeyInformation(0b01001110000010));
eapol::KeyFrameTx::new(
eapol::ProtocolVersion::IEEE802DOT1X2004,
key_frame_fields,
vec![],
test_util::mic_len(),
)
}
#[test]
fn zeroed_iv_8021x2004() {
let mut key_frame = fake_key_frame();
key_frame.protocol_version = eapol::ProtocolVersion::IEEE802DOT1X2004;
let key_frame = key_frame
.serialize()
.finalize_with_mic(&vec![0u8; test_util::mic_len()][..])
.expect("failed to construct key frame");
verify_group_key_frame(key_frame, Role::Supplicant);
}
#[test]
fn random_iv_8021x2004() {
let mut key_frame = fake_key_frame();
key_frame.protocol_version = eapol::ProtocolVersion::IEEE802DOT1X2004;
key_frame.key_frame_fields.key_iv.copy_from_slice(&[1; 16]);
let key_frame = key_frame
.serialize()
.finalize_with_mic(&vec![0u8; test_util::mic_len()][..])
.expect("failed to construct key frame");
verify_group_key_frame(key_frame, Role::Supplicant);
}
#[test]
fn zeroed_iv_8021x2001() {
let mut key_frame = fake_key_frame();
key_frame.protocol_version = eapol::ProtocolVersion::IEEE802DOT1X2001;
let key_frame = key_frame
.serialize()
.finalize_with_mic(&vec![0u8; test_util::mic_len()][..])
.expect("failed to construct key frame");
verify_group_key_frame(key_frame, Role::Supplicant);
}
#[test]
fn random_iv_8021x2001() {
let mut key_frame = fake_key_frame();
key_frame.protocol_version = eapol::ProtocolVersion::IEEE802DOT1X2001;
key_frame.key_frame_fields.key_iv.copy_from_slice(&[1; 16]);
let key_frame = key_frame
.serialize()
.finalize_with_mic(&vec![0u8; test_util::mic_len()][..])
.expect("failed to construct key frame");
verify_group_key_frame(key_frame, Role::Supplicant);
}
}