mirror of
https://git.kernel.org/pub/scm/network/wireless/iwd.git
synced 2025-01-10 17:52:36 +01:00
netdev: FT version of association messages
If an MD IE is supplied to netdev_connect, pass that MD IE in the associate request, then validate and handle the MD IE and FT IE in the associate response from AP.
This commit is contained in:
parent
955ba74d2d
commit
a35e0c2690
@ -737,7 +737,7 @@ void device_connect_network(struct device *device, struct network *network,
|
|||||||
|
|
||||||
device->connect_pending = l_dbus_message_ref(message);
|
device->connect_pending = l_dbus_message_ref(message);
|
||||||
|
|
||||||
if (netdev_connect(device->netdev, bss, sm,
|
if (netdev_connect(device->netdev, bss, sm, NULL,
|
||||||
device_netdev_event,
|
device_netdev_event,
|
||||||
device_connect_cb, device) < 0) {
|
device_connect_cb, device) < 0) {
|
||||||
if (sm)
|
if (sm)
|
||||||
|
121
src/netdev.c
121
src/netdev.c
@ -71,6 +71,7 @@ struct netdev {
|
|||||||
uint32_t connect_cmd_id;
|
uint32_t connect_cmd_id;
|
||||||
uint32_t disconnect_cmd_id;
|
uint32_t disconnect_cmd_id;
|
||||||
enum netdev_result result;
|
enum netdev_result result;
|
||||||
|
uint8_t *mde;
|
||||||
|
|
||||||
struct l_queue *watches;
|
struct l_queue *watches;
|
||||||
uint32_t next_watch_id;
|
uint32_t next_watch_id;
|
||||||
@ -317,6 +318,9 @@ static void netdev_connect_free(struct netdev *netdev)
|
|||||||
l_genl_family_cancel(nl80211, netdev->disconnect_cmd_id);
|
l_genl_family_cancel(nl80211, netdev->disconnect_cmd_id);
|
||||||
netdev->disconnect_cmd_id = 0;
|
netdev->disconnect_cmd_id = 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
l_free(netdev->mde);
|
||||||
|
netdev->mde = NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
static void netdev_connect_failed(struct l_genl_msg *msg, void *user_data)
|
static void netdev_connect_failed(struct l_genl_msg *msg, void *user_data)
|
||||||
@ -1060,6 +1064,8 @@ static void netdev_connect_event(struct l_genl_msg *msg,
|
|||||||
uint16_t type, len;
|
uint16_t type, len;
|
||||||
const void *data;
|
const void *data;
|
||||||
const uint16_t *status_code = NULL;
|
const uint16_t *status_code = NULL;
|
||||||
|
const uint8_t *ies = NULL;
|
||||||
|
size_t ies_len;
|
||||||
|
|
||||||
l_debug("");
|
l_debug("");
|
||||||
|
|
||||||
@ -1081,6 +1087,10 @@ static void netdev_connect_event(struct l_genl_msg *msg,
|
|||||||
status_code = data;
|
status_code = data;
|
||||||
|
|
||||||
break;
|
break;
|
||||||
|
case NL80211_ATTR_RESP_IE:
|
||||||
|
ies = data;
|
||||||
|
ies_len = len;
|
||||||
|
break;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -1088,6 +1098,82 @@ static void netdev_connect_event(struct l_genl_msg *msg,
|
|||||||
if (!status_code || *status_code != 0)
|
if (!status_code || *status_code != 0)
|
||||||
goto error;
|
goto error;
|
||||||
|
|
||||||
|
/* Check 802.11r IEs */
|
||||||
|
if (netdev->mde && !ies)
|
||||||
|
goto error;
|
||||||
|
|
||||||
|
if (ies) {
|
||||||
|
struct ie_tlv_iter iter;
|
||||||
|
const uint8_t *mde = NULL;
|
||||||
|
const uint8_t *fte = NULL;
|
||||||
|
|
||||||
|
ie_tlv_iter_init(&iter, ies, ies_len);
|
||||||
|
|
||||||
|
while (ie_tlv_iter_next(&iter)) {
|
||||||
|
switch (ie_tlv_iter_get_tag(&iter)) {
|
||||||
|
case IE_TYPE_MOBILITY_DOMAIN:
|
||||||
|
if (mde)
|
||||||
|
goto error;
|
||||||
|
|
||||||
|
mde = ie_tlv_iter_get_data(&iter) - 2;
|
||||||
|
break;
|
||||||
|
|
||||||
|
case IE_TYPE_FAST_BSS_TRANSITION:
|
||||||
|
if (fte)
|
||||||
|
goto error;
|
||||||
|
|
||||||
|
fte = ie_tlv_iter_get_data(&iter) - 2;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/* An MD IE identical to the one we sent must be present */
|
||||||
|
if (netdev->mde && (!mde || memcmp(netdev->mde,
|
||||||
|
mde, netdev->mde[1] + 2)))
|
||||||
|
goto error;
|
||||||
|
|
||||||
|
/*
|
||||||
|
* An FT IE is required in an initial mobility domain
|
||||||
|
* association and re-associations in an RSN but not present
|
||||||
|
* in a non-RSN (12.4.2 vs. 12.4.3).
|
||||||
|
*/
|
||||||
|
|
||||||
|
if (netdev->mde && !netdev->sm && fte)
|
||||||
|
goto error;
|
||||||
|
|
||||||
|
if (netdev->mde && netdev->sm) {
|
||||||
|
struct ie_ft_info ft_info;
|
||||||
|
uint8_t zeros[32];
|
||||||
|
|
||||||
|
if (!fte)
|
||||||
|
goto error;
|
||||||
|
|
||||||
|
eapol_sm_set_fte(netdev->sm, fte);
|
||||||
|
|
||||||
|
if (ie_parse_fast_bss_transition_from_data(fte,
|
||||||
|
fte[1] + 2,
|
||||||
|
&ft_info) < 0)
|
||||||
|
goto error;
|
||||||
|
|
||||||
|
/*
|
||||||
|
* 12.4.2: "The FTE shall have a MIC information
|
||||||
|
* element count of zero (i.e., no MIC present)
|
||||||
|
* and have ANonce, SNonce, and MIC fields set to 0."
|
||||||
|
*/
|
||||||
|
memset(zeros, 0, 32);
|
||||||
|
|
||||||
|
if (ft_info.mic_element_count != 0 ||
|
||||||
|
memcmp(ft_info.mic, zeros, 16) ||
|
||||||
|
memcmp(ft_info.anonce, zeros, 32) ||
|
||||||
|
memcmp(ft_info.snonce, zeros, 32))
|
||||||
|
goto error;
|
||||||
|
|
||||||
|
eapol_sm_set_kh_ids(netdev->sm, ft_info.r0khid,
|
||||||
|
ft_info.r0khid_len,
|
||||||
|
ft_info.r1khid);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
if (netdev->eapol_active) {
|
if (netdev->eapol_active) {
|
||||||
/*
|
/*
|
||||||
* Start processing EAPoL frames now that the state machine
|
* Start processing EAPoL frames now that the state machine
|
||||||
@ -1157,10 +1243,13 @@ static void netdev_cmd_connect_cb(struct l_genl_msg *msg, void *user_data)
|
|||||||
|
|
||||||
static struct l_genl_msg *netdev_build_cmd_connect(struct netdev *netdev,
|
static struct l_genl_msg *netdev_build_cmd_connect(struct netdev *netdev,
|
||||||
struct scan_bss *bss,
|
struct scan_bss *bss,
|
||||||
struct eapol_sm *sm)
|
struct eapol_sm *sm,
|
||||||
|
const uint8_t *mde)
|
||||||
{
|
{
|
||||||
uint32_t auth_type = NL80211_AUTHTYPE_OPEN_SYSTEM;
|
uint32_t auth_type = NL80211_AUTHTYPE_OPEN_SYSTEM;
|
||||||
struct l_genl_msg *msg;
|
struct l_genl_msg *msg;
|
||||||
|
struct iovec iov[2];
|
||||||
|
int iov_elems = 0;
|
||||||
|
|
||||||
msg = l_genl_msg_new_sized(NL80211_CMD_CONNECT, 512);
|
msg = l_genl_msg_new_sized(NL80211_CMD_CONNECT, 512);
|
||||||
l_genl_msg_append_attr(msg, NL80211_ATTR_IFINDEX, 4, &netdev->index);
|
l_genl_msg_append_attr(msg, NL80211_ATTR_IFINDEX, 4, &netdev->index);
|
||||||
@ -1201,10 +1290,21 @@ static struct l_genl_msg *netdev_build_cmd_connect(struct netdev *netdev,
|
|||||||
l_genl_msg_append_attr(msg, NL80211_ATTR_CONTROL_PORT, 0, NULL);
|
l_genl_msg_append_attr(msg, NL80211_ATTR_CONTROL_PORT, 0, NULL);
|
||||||
|
|
||||||
ie = eapol_sm_get_own_ie(sm, &ie_len);
|
ie = eapol_sm_get_own_ie(sm, &ie_len);
|
||||||
if (ie)
|
if (ie) {
|
||||||
l_genl_msg_append_attr(msg, NL80211_ATTR_IE,
|
iov[iov_elems].iov_base = (void *) ie;
|
||||||
ie_len, ie);
|
iov[iov_elems].iov_len = ie_len;
|
||||||
|
iov_elems += 1;
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (mde) {
|
||||||
|
iov[iov_elems].iov_base = (void *) mde;
|
||||||
|
iov[iov_elems].iov_len = mde[1] + 2;
|
||||||
|
iov_elems += 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (iov_elems)
|
||||||
|
l_genl_msg_append_attrv(msg, NL80211_ATTR_IE, iov, iov_elems);
|
||||||
|
|
||||||
return msg;
|
return msg;
|
||||||
}
|
}
|
||||||
@ -1213,6 +1313,7 @@ static int netdev_connect_common(struct netdev *netdev,
|
|||||||
struct l_genl_msg *cmd_connect,
|
struct l_genl_msg *cmd_connect,
|
||||||
struct scan_bss *bss,
|
struct scan_bss *bss,
|
||||||
struct eapol_sm *sm,
|
struct eapol_sm *sm,
|
||||||
|
const uint8_t *mde,
|
||||||
netdev_event_func_t event_filter,
|
netdev_event_func_t event_filter,
|
||||||
netdev_connect_cb_t cb, void *user_data)
|
netdev_connect_cb_t cb, void *user_data)
|
||||||
{
|
{
|
||||||
@ -1232,12 +1333,16 @@ static int netdev_connect_common(struct netdev *netdev,
|
|||||||
netdev->connected = true;
|
netdev->connected = true;
|
||||||
netdev->sm = sm;
|
netdev->sm = sm;
|
||||||
|
|
||||||
|
if (mde)
|
||||||
|
netdev->mde = l_memdup(mde, mde[1] + 2);
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
int netdev_connect(struct netdev *netdev, struct scan_bss *bss,
|
int netdev_connect(struct netdev *netdev, struct scan_bss *bss,
|
||||||
struct eapol_sm *sm,
|
struct eapol_sm *sm,
|
||||||
|
const uint8_t *mde,
|
||||||
netdev_event_func_t event_filter,
|
netdev_event_func_t event_filter,
|
||||||
netdev_connect_cb_t cb, void *user_data)
|
netdev_connect_cb_t cb, void *user_data)
|
||||||
{
|
{
|
||||||
@ -1246,11 +1351,11 @@ int netdev_connect(struct netdev *netdev, struct scan_bss *bss,
|
|||||||
if (netdev->connected)
|
if (netdev->connected)
|
||||||
return -EISCONN;
|
return -EISCONN;
|
||||||
|
|
||||||
cmd_connect = netdev_build_cmd_connect(netdev, bss, sm);
|
cmd_connect = netdev_build_cmd_connect(netdev, bss, sm, mde);
|
||||||
if (!cmd_connect)
|
if (!cmd_connect)
|
||||||
return -EINVAL;
|
return -EINVAL;
|
||||||
|
|
||||||
return netdev_connect_common(netdev, cmd_connect, bss, sm,
|
return netdev_connect_common(netdev, cmd_connect, bss, sm, mde,
|
||||||
event_filter, cb, user_data);
|
event_filter, cb, user_data);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -1269,7 +1374,7 @@ int netdev_connect_wsc(struct netdev *netdev, struct scan_bss *bss,
|
|||||||
if (netdev->connected)
|
if (netdev->connected)
|
||||||
return -EISCONN;
|
return -EISCONN;
|
||||||
|
|
||||||
cmd_connect = netdev_build_cmd_connect(netdev, bss, NULL);
|
cmd_connect = netdev_build_cmd_connect(netdev, bss, NULL, NULL);
|
||||||
if (!cmd_connect)
|
if (!cmd_connect)
|
||||||
return -EINVAL;
|
return -EINVAL;
|
||||||
|
|
||||||
@ -1289,7 +1394,7 @@ int netdev_connect_wsc(struct netdev *netdev, struct scan_bss *bss,
|
|||||||
l_genl_msg_append_attr(cmd_connect, NL80211_ATTR_IE, ie_len, ie);
|
l_genl_msg_append_attr(cmd_connect, NL80211_ATTR_IE, ie_len, ie);
|
||||||
l_free(ie);
|
l_free(ie);
|
||||||
|
|
||||||
return netdev_connect_common(netdev, cmd_connect, bss, sm,
|
return netdev_connect_common(netdev, cmd_connect, bss, sm, NULL,
|
||||||
event_filter, cb, user_data);
|
event_filter, cb, user_data);
|
||||||
|
|
||||||
error:
|
error:
|
||||||
|
@ -73,7 +73,7 @@ const char *netdev_get_name(struct netdev *netdev);
|
|||||||
bool netdev_get_is_up(struct netdev *netdev);
|
bool netdev_get_is_up(struct netdev *netdev);
|
||||||
|
|
||||||
int netdev_connect(struct netdev *netdev, struct scan_bss *bss,
|
int netdev_connect(struct netdev *netdev, struct scan_bss *bss,
|
||||||
struct eapol_sm *sm,
|
struct eapol_sm *sm, const uint8_t *mde,
|
||||||
netdev_event_func_t event_filter,
|
netdev_event_func_t event_filter,
|
||||||
netdev_connect_cb_t cb, void *user_data);
|
netdev_connect_cb_t cb, void *user_data);
|
||||||
int netdev_connect_wsc(struct netdev *netdev, struct scan_bss *bss,
|
int netdev_connect_wsc(struct netdev *netdev, struct scan_bss *bss,
|
||||||
|
Loading…
Reference in New Issue
Block a user