|
@@ -19875,111 +19875,109 @@ static void wlan_hdd_cfg80211_clear_privacy(struct hdd_adapter *adapter)
|
|
|
wlan_hdd_clear_wapi_privacy(adapter);
|
|
|
}
|
|
|
|
|
|
-int wlan_hdd_try_disconnect(struct hdd_adapter *adapter)
|
|
|
+static int wlan_hdd_wait_for_disconnect(mac_handle_t mac_handle,
|
|
|
+ struct hdd_adapter *adapter,
|
|
|
+ uint16_t reason)
|
|
|
{
|
|
|
+ eConnectionState prev_conn_state;
|
|
|
+ struct hdd_station_ctx *sta_ctx = WLAN_HDD_GET_STATION_CTX_PTR(adapter);
|
|
|
+ QDF_STATUS status = QDF_STATUS_SUCCESS;
|
|
|
+ int ret = 0;
|
|
|
unsigned long rc;
|
|
|
- struct hdd_station_ctx *sta_ctx;
|
|
|
- int status, result = 0;
|
|
|
- mac_handle_t mac_handle;
|
|
|
uint32_t wait_time = SME_DISCONNECT_TIMEOUT;
|
|
|
- struct hdd_context *hdd_ctx;
|
|
|
|
|
|
- hdd_ctx = WLAN_HDD_GET_CTX(adapter);
|
|
|
- sta_ctx = WLAN_HDD_GET_STATION_CTX_PTR(adapter);
|
|
|
- mac_handle = hdd_adapter_get_mac_handle(adapter);
|
|
|
- if (adapter->device_mode == QDF_STA_MODE) {
|
|
|
- hdd_debug("Stop firmware roaming");
|
|
|
- sme_stop_roaming(mac_handle, adapter->vdev_id,
|
|
|
- REASON_DRIVER_DISABLED,
|
|
|
- RSO_INVALID_REQUESTOR);
|
|
|
+ /* If already in disconnecting state just wait for its completion */
|
|
|
+ if (sta_ctx->conn_info.conn_state == eConnectionState_Disconnecting)
|
|
|
+ goto wait_for_disconnect;
|
|
|
+
|
|
|
+ INIT_COMPLETION(adapter->disconnect_comp_var);
|
|
|
+ prev_conn_state = sta_ctx->conn_info.conn_state;
|
|
|
+ hdd_conn_set_connection_state(adapter, eConnectionState_Disconnecting);
|
|
|
+
|
|
|
+ status = sme_roam_disconnect(mac_handle, adapter->vdev_id, reason);
|
|
|
+ if (status == QDF_STATUS_CMD_NOT_QUEUED &&
|
|
|
+ prev_conn_state == eConnectionState_Connecting) {
|
|
|
/*
|
|
|
- * If firmware has already started roaming process, driver
|
|
|
- * needs to wait for processing of this disconnect request.
|
|
|
- *
|
|
|
+ * Wait here instead of returning directly, this will block the
|
|
|
+ * next connect command and allow processing of the scan for
|
|
|
+ * ssid and the previous connect command in CSR.
|
|
|
*/
|
|
|
- INIT_COMPLETION(adapter->roaming_comp_var);
|
|
|
- if (hdd_is_roaming_in_progress(hdd_ctx) ||
|
|
|
- sme_neighbor_middle_of_roaming(mac_handle,
|
|
|
- adapter->vdev_id)) {
|
|
|
- rc = wait_for_completion_timeout(
|
|
|
- &adapter->roaming_comp_var,
|
|
|
- msecs_to_jiffies(WLAN_WAIT_TIME_STOP_ROAM));
|
|
|
- if (!rc) {
|
|
|
- hdd_err("roaming_comp_var time out vdev id: %d",
|
|
|
- adapter->vdev_id);
|
|
|
- /* Clear roaming in progress flag */
|
|
|
- hdd_set_roaming_in_progress(false);
|
|
|
- }
|
|
|
- if (adapter->roam_ho_fail) {
|
|
|
- INIT_COMPLETION(adapter->disconnect_comp_var);
|
|
|
- hdd_conn_set_connection_state(adapter,
|
|
|
- eConnectionState_Disconnecting);
|
|
|
- }
|
|
|
- }
|
|
|
+ hdd_debug("CSR not connected but scan for SSID is in progress, wait for scan to be aborted or completed.");
|
|
|
+ } else if (QDF_IS_STATUS_ERROR(status)) {
|
|
|
+ hdd_debug("SB Disconnect in progress/SME is disconencted/Connect removed from pending queue: status = %d",
|
|
|
+ status);
|
|
|
+ /*
|
|
|
+ * Wait here instead of returning directly. This will block the
|
|
|
+ * next connect command and allow processing of the disconnect
|
|
|
+ * in SME. As disconnect is already in progress, wait here for
|
|
|
+ * WLAN_WAIT_DISCONNECT_ALREADY_IN_PROGRESS instead of
|
|
|
+ * SME_DISCONNECT_TIMEOUT.
|
|
|
+ */
|
|
|
+ wait_time = WLAN_WAIT_DISCONNECT_ALREADY_IN_PROGRESS;
|
|
|
}
|
|
|
|
|
|
- if ((QDF_IBSS_MODE == adapter->device_mode) ||
|
|
|
- (eConnectionState_Associated == sta_ctx->conn_info.conn_state) ||
|
|
|
- (eConnectionState_Connecting == sta_ctx->conn_info.conn_state) ||
|
|
|
- (eConnectionState_IbssConnected == sta_ctx->conn_info.conn_state)) {
|
|
|
- eConnectionState prev_conn_state;
|
|
|
-
|
|
|
- INIT_COMPLETION(adapter->disconnect_comp_var);
|
|
|
- prev_conn_state = sta_ctx->conn_info.conn_state;
|
|
|
- hdd_conn_set_connection_state(adapter,
|
|
|
- eConnectionState_Disconnecting);
|
|
|
- /* Issue disconnect to CSR */
|
|
|
- status = sme_roam_disconnect(mac_handle,
|
|
|
- adapter->vdev_id,
|
|
|
- eCSR_DISCONNECT_REASON_UNSPECIFIED);
|
|
|
-
|
|
|
- if ((status == QDF_STATUS_CMD_NOT_QUEUED) &&
|
|
|
- prev_conn_state != eConnectionState_Connecting) {
|
|
|
- hdd_debug("Already disconnect in progress");
|
|
|
- result = 0;
|
|
|
- /*
|
|
|
- * Wait here instead of returning directly. This will
|
|
|
- * block the connect command and allow processing
|
|
|
- * of the disconnect in SME. As disconnect is already
|
|
|
- * in progress, wait here for 1 sec instead of 5 sec.
|
|
|
- */
|
|
|
- wait_time = WLAN_WAIT_DISCONNECT_ALREADY_IN_PROGRESS;
|
|
|
- } else if (status == QDF_STATUS_CMD_NOT_QUEUED) {
|
|
|
- /*
|
|
|
- * Wait here instead of returning directly, this will
|
|
|
- * block the connect command and allow processing
|
|
|
- * of the scan for ssid and the previous connect command
|
|
|
- * in CSR.
|
|
|
- */
|
|
|
- hdd_debug("Already disconnected or connect was in sme/roam pending list and removed by disconnect");
|
|
|
- } else if (0 != status) {
|
|
|
- hdd_err("sme_roam_disconnect failure, status: %d",
|
|
|
- (int)status);
|
|
|
- sta_ctx->sta_debug_state = status;
|
|
|
- result = -EINVAL;
|
|
|
- goto disconnected;
|
|
|
- }
|
|
|
+wait_for_disconnect:
|
|
|
+ rc = wait_for_completion_timeout(&adapter->disconnect_comp_var,
|
|
|
+ msecs_to_jiffies(wait_time));
|
|
|
|
|
|
- rc = wait_for_completion_timeout(&adapter->disconnect_comp_var,
|
|
|
- msecs_to_jiffies(wait_time));
|
|
|
- if (!rc && (QDF_STATUS_CMD_NOT_QUEUED != status)) {
|
|
|
- hdd_err("Sme disconnect event timed out vdev id: %d sta_debug_state: %d",
|
|
|
- adapter->vdev_id, sta_ctx->sta_debug_state);
|
|
|
- result = -ETIMEDOUT;
|
|
|
- }
|
|
|
- } else if (eConnectionState_Disconnecting ==
|
|
|
- sta_ctx->conn_info.conn_state) {
|
|
|
- rc = wait_for_completion_timeout(&adapter->disconnect_comp_var,
|
|
|
- msecs_to_jiffies(wait_time));
|
|
|
+ if (!rc && QDF_IS_STATUS_SUCCESS(status)) {
|
|
|
+ hdd_err("Disconnect timed out!!!");
|
|
|
+ ret = -ETIMEDOUT;
|
|
|
+ }
|
|
|
+
|
|
|
+ hdd_conn_set_connection_state(adapter, eConnectionState_NotConnected);
|
|
|
+
|
|
|
+ return ret;
|
|
|
+}
|
|
|
+
|
|
|
+static void wlan_hdd_wait_for_roaming(mac_handle_t mac_handle,
|
|
|
+ struct hdd_adapter *adapter)
|
|
|
+{
|
|
|
+ struct hdd_context *hdd_ctx;
|
|
|
+ unsigned long rc;
|
|
|
+
|
|
|
+ if (adapter->device_mode != QDF_STA_MODE)
|
|
|
+ return;
|
|
|
+
|
|
|
+ hdd_ctx = WLAN_HDD_GET_CTX(adapter);
|
|
|
+ hdd_debug("Stop firmware roaming");
|
|
|
+ sme_stop_roaming(mac_handle, adapter->vdev_id,
|
|
|
+ REASON_DRIVER_DISABLED,
|
|
|
+ RSO_INVALID_REQUESTOR);
|
|
|
+ /*
|
|
|
+ * If firmware has already started roaming process, driver
|
|
|
+ * needs to wait for processing of this disconnect request.
|
|
|
+ *
|
|
|
+ */
|
|
|
+ INIT_COMPLETION(adapter->roaming_comp_var);
|
|
|
+ if (hdd_is_roaming_in_progress(hdd_ctx) ||
|
|
|
+ sme_neighbor_middle_of_roaming(mac_handle,
|
|
|
+ adapter->vdev_id)) {
|
|
|
+ rc = wait_for_completion_timeout(&adapter->roaming_comp_var,
|
|
|
+ msecs_to_jiffies(WLAN_WAIT_TIME_STOP_ROAM));
|
|
|
if (!rc) {
|
|
|
- hdd_err("Disconnect event timed out vdev id: %d sta_debug_state: %d",
|
|
|
- adapter->vdev_id, sta_ctx->sta_debug_state);
|
|
|
- result = -ETIMEDOUT;
|
|
|
+ hdd_err("roaming_comp_var time out vdev id: %d",
|
|
|
+ adapter->vdev_id);
|
|
|
+ /* Clear roaming in progress flag */
|
|
|
+ hdd_set_roaming_in_progress(false);
|
|
|
+ }
|
|
|
+ if (adapter->roam_ho_fail) {
|
|
|
+ INIT_COMPLETION(adapter->disconnect_comp_var);
|
|
|
+ hdd_conn_set_connection_state(adapter,
|
|
|
+ eConnectionState_Disconnecting);
|
|
|
}
|
|
|
}
|
|
|
-disconnected:
|
|
|
- hdd_conn_set_connection_state(adapter, eConnectionState_NotConnected);
|
|
|
- return result;
|
|
|
+}
|
|
|
+
|
|
|
+int wlan_hdd_try_disconnect(struct hdd_adapter *adapter)
|
|
|
+{
|
|
|
+ mac_handle_t mac_handle;
|
|
|
+
|
|
|
+ mac_handle = hdd_adapter_get_mac_handle(adapter);
|
|
|
+ wlan_hdd_wait_for_roaming(mac_handle, adapter);
|
|
|
+
|
|
|
+ return wlan_hdd_wait_for_disconnect(mac_handle, adapter,
|
|
|
+ eCSR_DISCONNECT_REASON_UNSPECIFIED);
|
|
|
}
|
|
|
|
|
|
/**
|
|
@@ -20336,96 +20334,19 @@ static int wlan_hdd_cfg80211_connect(struct wiphy *wiphy,
|
|
|
|
|
|
int wlan_hdd_disconnect(struct hdd_adapter *adapter, u16 reason)
|
|
|
{
|
|
|
- QDF_STATUS status;
|
|
|
- int result = 0;
|
|
|
- unsigned long rc;
|
|
|
- struct hdd_station_ctx *sta_ctx = WLAN_HDD_GET_STATION_CTX_PTR(adapter);
|
|
|
- struct hdd_context *hdd_ctx = WLAN_HDD_GET_CTX(adapter);
|
|
|
- eConnectionState prev_conn_state;
|
|
|
+ int ret;
|
|
|
mac_handle_t mac_handle;
|
|
|
- uint32_t wait_time = SME_DISCONNECT_TIMEOUT;
|
|
|
|
|
|
- hdd_enter();
|
|
|
-
|
|
|
- mac_handle = hdd_ctx->mac_handle;
|
|
|
- if (adapter->device_mode == QDF_STA_MODE) {
|
|
|
- hdd_debug("Stop firmware roaming");
|
|
|
- status = sme_stop_roaming(mac_handle, adapter->vdev_id,
|
|
|
- REASON_DRIVER_DISABLED,
|
|
|
- RSO_INVALID_REQUESTOR);
|
|
|
- /*
|
|
|
- * If firmware has already started roaming process, driver
|
|
|
- * needs to wait for processing of this disconnect request.
|
|
|
- *
|
|
|
- */
|
|
|
- INIT_COMPLETION(adapter->roaming_comp_var);
|
|
|
- if (hdd_is_roaming_in_progress(hdd_ctx) ||
|
|
|
- sme_neighbor_middle_of_roaming(mac_handle,
|
|
|
- adapter->vdev_id)) {
|
|
|
- rc = wait_for_completion_timeout(
|
|
|
- &adapter->roaming_comp_var,
|
|
|
- msecs_to_jiffies(WLAN_WAIT_TIME_STOP_ROAM));
|
|
|
- if (!rc) {
|
|
|
- hdd_err("roaming_comp_var time out vdev id: %d",
|
|
|
- adapter->vdev_id);
|
|
|
- /* Clear roaming in progress flag */
|
|
|
- hdd_set_roaming_in_progress(false);
|
|
|
- }
|
|
|
- if (adapter->roam_ho_fail) {
|
|
|
- INIT_COMPLETION(adapter->disconnect_comp_var);
|
|
|
- hdd_debug("Disabling queues");
|
|
|
- wlan_hdd_netif_queue_control(adapter,
|
|
|
- WLAN_STOP_ALL_NETIF_QUEUE_N_CARRIER,
|
|
|
- WLAN_CONTROL_PATH);
|
|
|
- hdd_conn_set_connection_state(adapter,
|
|
|
- eConnectionState_Disconnecting);
|
|
|
- goto wait_for_disconnect;
|
|
|
- }
|
|
|
- }
|
|
|
- }
|
|
|
+ mac_handle = hdd_adapter_get_mac_handle(adapter);
|
|
|
+ wlan_hdd_wait_for_roaming(mac_handle, adapter);
|
|
|
|
|
|
- INIT_COMPLETION(adapter->disconnect_comp_var);
|
|
|
- prev_conn_state = sta_ctx->conn_info.conn_state;
|
|
|
/*stop tx queues */
|
|
|
hdd_info("Disabling queues");
|
|
|
wlan_hdd_netif_queue_control(adapter,
|
|
|
WLAN_STOP_ALL_NETIF_QUEUE_N_CARRIER, WLAN_CONTROL_PATH);
|
|
|
- hdd_debug("Set HDD conn_state to eConnectionState_Disconnecting");
|
|
|
- hdd_conn_set_connection_state(adapter, eConnectionState_Disconnecting);
|
|
|
|
|
|
- /* issue disconnect */
|
|
|
+ ret = wlan_hdd_wait_for_disconnect(mac_handle, adapter, reason);
|
|
|
|
|
|
- status = sme_roam_disconnect(mac_handle,
|
|
|
- adapter->vdev_id, reason);
|
|
|
- if (QDF_STATUS_CMD_NOT_QUEUED == status &&
|
|
|
- prev_conn_state == eConnectionState_Connecting) {
|
|
|
- /*
|
|
|
- * Wait here instead of returning directly, this will block the
|
|
|
- * next connect command and allow processing of the scan for
|
|
|
- * ssid and the previous connect command in CSR.
|
|
|
- */
|
|
|
- hdd_debug("CSR is not in connected state but scan for SSID is in progress, wait for scan to be aborted or completed.");
|
|
|
- } else if (QDF_IS_STATUS_ERROR(status)) {
|
|
|
- hdd_debug("status = %d, already disconnect in progress or SME is disconencted OR connect removed from pending queue",
|
|
|
- status);
|
|
|
- /*
|
|
|
- * Wait here instead of returning directly. This will block the
|
|
|
- * next connect command and allow processing of the disconnect
|
|
|
- * in SME. As disconnect is already in progress, wait here for
|
|
|
- * WLAN_WAIT_DISCONNECT_ALREADY_IN_PROGRESS instead of
|
|
|
- * SME_DISCONNECT_TIMEOUT.
|
|
|
- */
|
|
|
- wait_time = WLAN_WAIT_DISCONNECT_ALREADY_IN_PROGRESS;
|
|
|
- }
|
|
|
-wait_for_disconnect:
|
|
|
- rc = wait_for_completion_timeout(&adapter->disconnect_comp_var,
|
|
|
- msecs_to_jiffies(wait_time));
|
|
|
-
|
|
|
- if (!rc && (QDF_STATUS_CMD_NOT_QUEUED != status)) {
|
|
|
- hdd_err("Failed to disconnect, timed out");
|
|
|
- result = -ETIMEDOUT;
|
|
|
- }
|
|
|
- hdd_conn_set_connection_state(adapter, eConnectionState_NotConnected);
|
|
|
#if LINUX_VERSION_CODE >= KERNEL_VERSION(3, 11, 0)
|
|
|
/* Sending disconnect event to userspace for kernel version < 3.11
|
|
|
* is handled by __cfg80211_disconnect call to __cfg80211_disconnected
|
|
@@ -20435,7 +20356,7 @@ wait_for_disconnect:
|
|
|
WLAN_REASON_UNSPECIFIED, NULL, 0);
|
|
|
#endif
|
|
|
|
|
|
- return result;
|
|
|
+ return ret;
|
|
|
}
|
|
|
|
|
|
/**
|