|
@@ -0,0 +1,902 @@
|
|
|
+/*
|
|
|
+ * Copyright (c) 2017-2018 The Linux Foundation. All rights reserved.
|
|
|
+ *
|
|
|
+ * Permission to use, copy, modify, and/or distribute this software for
|
|
|
+ * any purpose with or without fee is hereby granted, provided that the
|
|
|
+ * above copyright notice and this permission notice appear in all
|
|
|
+ * copies.
|
|
|
+ *
|
|
|
+ * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL
|
|
|
+ * WARRANTIES WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED
|
|
|
+ * WARRANTIES OF MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE
|
|
|
+ * AUTHOR BE LIABLE FOR ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL
|
|
|
+ * DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR
|
|
|
+ * PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER
|
|
|
+ * TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR
|
|
|
+ * PERFORMANCE OF THIS SOFTWARE.
|
|
|
+ */
|
|
|
+/**
|
|
|
+ * DOC: wlan_serialization_internal.c
|
|
|
+ * This file defines the functions which are called
|
|
|
+ * from serialization public API's and are internal
|
|
|
+ * to serialization.
|
|
|
+ */
|
|
|
+
|
|
|
+#include <wlan_objmgr_vdev_obj.h>
|
|
|
+#include <wlan_objmgr_pdev_obj.h>
|
|
|
+#include <wlan_objmgr_psoc_obj.h>
|
|
|
+#include <qdf_list.h>
|
|
|
+#include <qdf_status.h>
|
|
|
+#include <wlan_utility.h>
|
|
|
+#include "wlan_serialization_api.h"
|
|
|
+#include "wlan_serialization_main_i.h"
|
|
|
+#include "wlan_serialization_utils_i.h"
|
|
|
+#include "wlan_serialization_non_scan_i.h"
|
|
|
+#include "wlan_serialization_scan_i.h"
|
|
|
+#include "wlan_serialization_internal_i.h"
|
|
|
+
|
|
|
+bool wlan_serialization_is_cmd_present_queue(
|
|
|
+ struct wlan_serialization_command *cmd,
|
|
|
+ uint8_t is_active_queue)
|
|
|
+{
|
|
|
+ qdf_list_t *queue;
|
|
|
+ bool status = false;
|
|
|
+ enum wlan_serialization_node node_type;
|
|
|
+ struct wlan_ser_pdev_obj *ser_pdev_obj;
|
|
|
+ struct wlan_ser_vdev_obj *ser_vdev_obj;
|
|
|
+ enum wlan_serialization_cmd_type cmd_type;
|
|
|
+
|
|
|
+ if (!cmd) {
|
|
|
+ ser_err("invalid cmd");
|
|
|
+ goto error;
|
|
|
+ }
|
|
|
+
|
|
|
+ cmd_type = cmd->cmd_type;
|
|
|
+
|
|
|
+ ser_pdev_obj = wlan_serialization_get_pdev_obj(
|
|
|
+ wlan_serialization_get_pdev_from_cmd(cmd));
|
|
|
+
|
|
|
+ if (!ser_pdev_obj) {
|
|
|
+ ser_err("invalid ser vdev obj");
|
|
|
+ goto error;
|
|
|
+ }
|
|
|
+
|
|
|
+ ser_vdev_obj = wlan_serialization_get_vdev_obj(
|
|
|
+ wlan_serialization_get_vdev_from_cmd(cmd));
|
|
|
+ if (!ser_vdev_obj) {
|
|
|
+ ser_err("invalid ser pdev obj");
|
|
|
+ goto error;
|
|
|
+ }
|
|
|
+
|
|
|
+ if (cmd_type < WLAN_SER_CMD_NONSCAN) {
|
|
|
+ queue = wlan_serialization_get_list_from_pdev_queue(
|
|
|
+ ser_pdev_obj, cmd_type, is_active_queue);
|
|
|
+ node_type = WLAN_SER_PDEV_NODE;
|
|
|
+ } else {
|
|
|
+ queue = wlan_serialization_get_list_from_vdev_queue(
|
|
|
+ ser_vdev_obj, cmd_type, is_active_queue);
|
|
|
+ node_type = WLAN_SER_VDEV_NODE;
|
|
|
+ }
|
|
|
+
|
|
|
+ status = wlan_serialization_is_cmd_present_in_given_queue(queue, cmd,
|
|
|
+ node_type);
|
|
|
+
|
|
|
+error:
|
|
|
+ return status;
|
|
|
+}
|
|
|
+
|
|
|
+enum wlan_serialization_status
|
|
|
+wlan_serialization_enqueue_cmd(struct wlan_serialization_command *cmd)
|
|
|
+{
|
|
|
+ enum wlan_serialization_status status = WLAN_SER_CMD_DENIED_UNSPECIFIED;
|
|
|
+ struct wlan_serialization_command_list *cmd_list;
|
|
|
+ qdf_list_node_t *nnode;
|
|
|
+ struct wlan_objmgr_pdev *pdev;
|
|
|
+ struct wlan_ser_pdev_obj *ser_pdev_obj;
|
|
|
+ struct wlan_serialization_pdev_queue *pdev_queue;
|
|
|
+ bool active_queue;
|
|
|
+
|
|
|
+ /* Enqueue process
|
|
|
+ * 1) peek through command structure and see what is the command type
|
|
|
+ * 2) two main types of commands to process
|
|
|
+ * a) SCAN
|
|
|
+ * b) NON-SCAN
|
|
|
+ * 3) for each command there are separate command queues per pdev
|
|
|
+ * 4) pull pdev from vdev structure and get the command queue associated
|
|
|
+ * with that pdev and try to enqueue on those queue
|
|
|
+ * 5) Thumb rule:
|
|
|
+ * a) There could be only 1 active non-scan command at a
|
|
|
+ * time including all total non-scan commands of all pdevs.
|
|
|
+ *
|
|
|
+ * example: pdev1 has 1 non-scan active command and
|
|
|
+ * pdev2 got 1 non-scan command then that command should go to
|
|
|
+ * pdev2's pending queue
|
|
|
+ *
|
|
|
+ * b) There could be only N number of scan commands at a time
|
|
|
+ * including all total scan commands of all pdevs
|
|
|
+ *
|
|
|
+ * example: Let's say N=8,
|
|
|
+ * pdev1's vdev1 has 5 scan command, pdev2's vdev1 has 3
|
|
|
+ * scan commands, if we get scan request on vdev2 then it will go
|
|
|
+ * to pending queue of vdev2 as we reached max allowed scan active
|
|
|
+ * command.
|
|
|
+ */
|
|
|
+
|
|
|
+ ser_enter();
|
|
|
+
|
|
|
+ if (!cmd) {
|
|
|
+ ser_err("NULL command");
|
|
|
+ goto error;
|
|
|
+ }
|
|
|
+
|
|
|
+ if (!cmd->cmd_cb) {
|
|
|
+ ser_err("no cmd_cb for cmd type:%d, id: %d",
|
|
|
+ cmd->cmd_type,
|
|
|
+ cmd->cmd_id);
|
|
|
+ goto error;
|
|
|
+ }
|
|
|
+
|
|
|
+ pdev = wlan_serialization_get_pdev_from_cmd(cmd);
|
|
|
+ if (!pdev) {
|
|
|
+ ser_err("pdev is invalid");
|
|
|
+ goto error;
|
|
|
+ }
|
|
|
+
|
|
|
+ ser_pdev_obj =
|
|
|
+ wlan_objmgr_pdev_get_comp_private_obj(
|
|
|
+ pdev,
|
|
|
+ WLAN_UMAC_COMP_SERIALIZATION);
|
|
|
+ if (!ser_pdev_obj) {
|
|
|
+ ser_err("Invalid ser_pdev_obj");
|
|
|
+ goto error;
|
|
|
+ }
|
|
|
+
|
|
|
+ pdev_queue = wlan_serialization_get_pdev_queue_obj(ser_pdev_obj,
|
|
|
+ cmd->cmd_type);
|
|
|
+ if (!pdev_queue) {
|
|
|
+ ser_err("pdev_queue is invalid");
|
|
|
+ goto error;
|
|
|
+ }
|
|
|
+
|
|
|
+ ser_debug("enqueue cmd: type[%d] id[%d] high_priority[%d] blocking[%d]",
|
|
|
+ cmd->cmd_type,
|
|
|
+ cmd->cmd_id,
|
|
|
+ cmd->is_high_priority,
|
|
|
+ cmd->is_blocking);
|
|
|
+
|
|
|
+ wlan_serialization_acquire_lock(pdev_queue);
|
|
|
+
|
|
|
+ active_queue = wlan_serialization_is_active_cmd_allowed(cmd);
|
|
|
+
|
|
|
+ if (wlan_serialization_is_cmd_present_queue(cmd, active_queue)) {
|
|
|
+ wlan_serialization_release_lock(pdev_queue);
|
|
|
+ ser_err("duplicate command, can't enqueue");
|
|
|
+ goto error;
|
|
|
+ }
|
|
|
+
|
|
|
+ if (wlan_serialization_remove_front(
|
|
|
+ &pdev_queue->cmd_pool_list,
|
|
|
+ &nnode) != QDF_STATUS_SUCCESS) {
|
|
|
+ wlan_serialization_release_lock(pdev_queue);
|
|
|
+ ser_err("Failed to get cmd buffer from global pool");
|
|
|
+ goto error;
|
|
|
+ }
|
|
|
+
|
|
|
+ ser_debug("Global pool node: %pK", nnode);
|
|
|
+
|
|
|
+ cmd_list =
|
|
|
+ qdf_container_of(nnode,
|
|
|
+ struct wlan_serialization_command_list,
|
|
|
+ pdev_node);
|
|
|
+
|
|
|
+ qdf_mem_copy(&cmd_list->cmd, cmd,
|
|
|
+ sizeof(struct wlan_serialization_command));
|
|
|
+
|
|
|
+ if (cmd->cmd_type < WLAN_SER_CMD_NONSCAN) {
|
|
|
+ status = wlan_ser_add_scan_cmd(ser_pdev_obj,
|
|
|
+ cmd_list,
|
|
|
+ active_queue);
|
|
|
+ } else {
|
|
|
+ status = wlan_ser_add_non_scan_cmd(ser_pdev_obj,
|
|
|
+ cmd_list,
|
|
|
+ active_queue);
|
|
|
+ }
|
|
|
+
|
|
|
+ if (status != WLAN_SER_CMD_PENDING && status != WLAN_SER_CMD_ACTIVE) {
|
|
|
+ ser_err("Failed to add cmd to active/pending queue");
|
|
|
+ qdf_mem_zero(&cmd_list->cmd,
|
|
|
+ sizeof(struct wlan_serialization_command));
|
|
|
+ wlan_serialization_insert_back(
|
|
|
+ &pdev_queue->cmd_pool_list,
|
|
|
+ &cmd_list->pdev_node);
|
|
|
+ }
|
|
|
+
|
|
|
+ if (WLAN_SER_CMD_ACTIVE == status) {
|
|
|
+ qdf_atomic_set_bit(CMD_MARKED_FOR_ACTIVATION,
|
|
|
+ &cmd_list->cmd_in_use);
|
|
|
+ }
|
|
|
+
|
|
|
+ wlan_serialization_release_lock(pdev_queue);
|
|
|
+
|
|
|
+ if (WLAN_SER_CMD_ACTIVE == status)
|
|
|
+ wlan_serialization_activate_cmd(cmd_list,
|
|
|
+ ser_pdev_obj);
|
|
|
+
|
|
|
+error:
|
|
|
+ ser_exit();
|
|
|
+
|
|
|
+ return status;
|
|
|
+}
|
|
|
+
|
|
|
+QDF_STATUS
|
|
|
+wlan_serialization_activate_multiple_cmd(
|
|
|
+ struct wlan_ser_pdev_obj *ser_pdev_obj)
|
|
|
+{
|
|
|
+ struct wlan_serialization_pdev_queue *pdev_queue;
|
|
|
+ qdf_list_t *active_queue;
|
|
|
+ QDF_STATUS peek_status = QDF_STATUS_E_FAILURE;
|
|
|
+ struct wlan_serialization_command_list *active_cmd_list;
|
|
|
+ uint32_t qsize;
|
|
|
+ uint32_t vdev_id;
|
|
|
+ qdf_list_node_t *nnode = NULL;
|
|
|
+ QDF_STATUS status = QDF_STATUS_SUCCESS;
|
|
|
+ struct wlan_objmgr_psoc *psoc = NULL;
|
|
|
+
|
|
|
+ pdev_queue = &ser_pdev_obj->pdev_q[SER_PDEV_QUEUE_COMP_NON_SCAN];
|
|
|
+ active_queue = &pdev_queue->active_list;
|
|
|
+
|
|
|
+ qsize = wlan_serialization_list_size(active_queue);
|
|
|
+ while (qsize--) {
|
|
|
+ peek_status = wlan_serialization_get_cmd_from_queue(
|
|
|
+ active_queue, &nnode);
|
|
|
+
|
|
|
+ if (peek_status != QDF_STATUS_SUCCESS) {
|
|
|
+ ser_err("can't peek cmd");
|
|
|
+ break;
|
|
|
+ }
|
|
|
+
|
|
|
+ active_cmd_list = qdf_container_of(
|
|
|
+ nnode, struct wlan_serialization_command_list,
|
|
|
+ pdev_node);
|
|
|
+
|
|
|
+ if (!qdf_atomic_test_bit(CMD_MARKED_FOR_ACTIVATION,
|
|
|
+ &active_cmd_list->cmd_in_use)) {
|
|
|
+ continue;
|
|
|
+ }
|
|
|
+
|
|
|
+ /*
|
|
|
+ * Command is already pushed to active queue.
|
|
|
+ * Now start the timer.
|
|
|
+ */
|
|
|
+ psoc = wlan_vdev_get_psoc(active_cmd_list->cmd.vdev);
|
|
|
+ wlan_serialization_find_and_start_timer(psoc,
|
|
|
+ &active_cmd_list->cmd);
|
|
|
+
|
|
|
+ ser_debug("cmd cb: type[%d] id[%d] : reason: %s",
|
|
|
+ active_cmd_list->cmd.cmd_type,
|
|
|
+ active_cmd_list->cmd.cmd_id,
|
|
|
+ "WLAN_SER_CB_ACTIVATE_CMD");
|
|
|
+
|
|
|
+ status = active_cmd_list->cmd.cmd_cb(&active_cmd_list->cmd,
|
|
|
+ WLAN_SER_CB_ACTIVATE_CMD);
|
|
|
+
|
|
|
+ qdf_atomic_clear_bit(CMD_MARKED_FOR_ACTIVATION,
|
|
|
+ &active_cmd_list->cmd_in_use);
|
|
|
+
|
|
|
+ qdf_atomic_set_bit(CMD_IS_ACTIVE,
|
|
|
+ &active_cmd_list->cmd_in_use);
|
|
|
+
|
|
|
+ vdev_id = wlan_vdev_get_id(active_cmd_list->cmd.vdev);
|
|
|
+ pdev_queue->vdev_active_cmd_bitmap |= (1 << vdev_id);
|
|
|
+
|
|
|
+ if (active_cmd_list->cmd.is_blocking)
|
|
|
+ pdev_queue->blocking_cmd_active = 1;
|
|
|
+
|
|
|
+ if (QDF_IS_STATUS_ERROR(status))
|
|
|
+ wlan_serialization_dequeue_cmd(&active_cmd_list->cmd,
|
|
|
+ true);
|
|
|
+ }
|
|
|
+ return status;
|
|
|
+}
|
|
|
+
|
|
|
+QDF_STATUS wlan_serialization_activate_cmd(
|
|
|
+ struct wlan_serialization_command_list *cmd_list,
|
|
|
+ struct wlan_ser_pdev_obj *ser_pdev_obj)
|
|
|
+{
|
|
|
+ QDF_STATUS status = QDF_STATUS_E_FAILURE;
|
|
|
+ struct wlan_objmgr_psoc *psoc = NULL;
|
|
|
+
|
|
|
+ psoc = wlan_vdev_get_psoc(cmd_list->cmd.vdev);
|
|
|
+ if (!psoc) {
|
|
|
+ ser_err("invalid psoc");
|
|
|
+ goto error;
|
|
|
+ }
|
|
|
+
|
|
|
+ /*
|
|
|
+ * command is already pushed to active queue above
|
|
|
+ * now start the timer and notify requestor
|
|
|
+ */
|
|
|
+ wlan_serialization_find_and_start_timer(psoc, &cmd_list->cmd);
|
|
|
+ /*
|
|
|
+ * Remember that serialization module may send
|
|
|
+ * this callback in same context through which it
|
|
|
+ * received the serialization request. Due to which
|
|
|
+ * it is caller's responsibility to ensure acquiring
|
|
|
+ * and releasing its own lock appropriately.
|
|
|
+ */
|
|
|
+
|
|
|
+ ser_debug("cmd cb: type[%d] id[%d] : reason: %s",
|
|
|
+ cmd_list->cmd.cmd_type,
|
|
|
+ cmd_list->cmd.cmd_id,
|
|
|
+ "WLAN_SER_CB_ACTIVATE_CMD");
|
|
|
+
|
|
|
+ status = cmd_list->cmd.cmd_cb(&cmd_list->cmd,
|
|
|
+ WLAN_SER_CB_ACTIVATE_CMD);
|
|
|
+
|
|
|
+ qdf_atomic_clear_bit(CMD_MARKED_FOR_ACTIVATION,
|
|
|
+ &cmd_list->cmd_in_use);
|
|
|
+ qdf_atomic_set_bit(CMD_IS_ACTIVE,
|
|
|
+ &cmd_list->cmd_in_use);
|
|
|
+
|
|
|
+ if (QDF_IS_STATUS_ERROR(status))
|
|
|
+ wlan_serialization_dequeue_cmd(&cmd_list->cmd, true);
|
|
|
+
|
|
|
+error:
|
|
|
+ return status;
|
|
|
+}
|
|
|
+
|
|
|
+bool
|
|
|
+wlan_serialization_is_active_cmd_allowed(struct wlan_serialization_command *cmd)
|
|
|
+{
|
|
|
+ struct wlan_objmgr_pdev *pdev;
|
|
|
+ bool active_cmd_allowed = 0;
|
|
|
+
|
|
|
+ pdev = wlan_serialization_get_pdev_from_cmd(cmd);
|
|
|
+ if (!pdev) {
|
|
|
+ ser_err("NULL pdev");
|
|
|
+ goto error;
|
|
|
+ }
|
|
|
+
|
|
|
+ if (cmd->cmd_type < WLAN_SER_CMD_NONSCAN)
|
|
|
+ active_cmd_allowed =
|
|
|
+ wlan_serialization_is_active_scan_cmd_allowed(cmd);
|
|
|
+ else
|
|
|
+ active_cmd_allowed =
|
|
|
+ wlan_serialization_is_active_non_scan_cmd_allowed(cmd);
|
|
|
+
|
|
|
+ ser_debug("active cmd_type[%d] cmd_id[%d] allowed: %d",
|
|
|
+ cmd->cmd_type,
|
|
|
+ cmd->cmd_id,
|
|
|
+ active_cmd_allowed);
|
|
|
+
|
|
|
+error:
|
|
|
+ return active_cmd_allowed;
|
|
|
+}
|
|
|
+
|
|
|
+enum wlan_serialization_status
|
|
|
+wlan_serialization_move_pending_to_active(
|
|
|
+ enum wlan_serialization_cmd_type cmd_type,
|
|
|
+ struct wlan_serialization_command_list **pcmd_list,
|
|
|
+ struct wlan_ser_pdev_obj *ser_pdev_obj,
|
|
|
+ struct wlan_objmgr_vdev *vdev,
|
|
|
+ bool blocking_cmd_removed,
|
|
|
+ bool blocking_cmd_waiting)
|
|
|
+{
|
|
|
+ enum wlan_serialization_status status;
|
|
|
+ struct wlan_serialization_pdev_queue *pdev_queue;
|
|
|
+
|
|
|
+ if (cmd_type < WLAN_SER_CMD_NONSCAN) {
|
|
|
+ status =
|
|
|
+ wlan_ser_move_scan_pending_to_active(
|
|
|
+ pcmd_list,
|
|
|
+ ser_pdev_obj);
|
|
|
+ } else {
|
|
|
+ pdev_queue =
|
|
|
+ &ser_pdev_obj->pdev_q[SER_PDEV_QUEUE_COMP_NON_SCAN];
|
|
|
+
|
|
|
+ if (!blocking_cmd_removed && !blocking_cmd_waiting)
|
|
|
+ status =
|
|
|
+ wlan_ser_move_non_scan_pending_to_active(
|
|
|
+ pcmd_list,
|
|
|
+ ser_pdev_obj,
|
|
|
+ vdev);
|
|
|
+ else
|
|
|
+ status =
|
|
|
+ wlan_ser_move_multiple_non_scan_pending_to_active(
|
|
|
+ ser_pdev_obj);
|
|
|
+ }
|
|
|
+
|
|
|
+ return status;
|
|
|
+}
|
|
|
+
|
|
|
+enum wlan_serialization_cmd_status
|
|
|
+wlan_serialization_dequeue_cmd(struct wlan_serialization_command *cmd,
|
|
|
+ uint8_t active_cmd)
|
|
|
+{
|
|
|
+ enum wlan_serialization_cmd_status status =
|
|
|
+ WLAN_SER_CMD_NOT_FOUND;
|
|
|
+ enum wlan_serialization_status ser_status =
|
|
|
+ WLAN_SER_CMD_DENIED_UNSPECIFIED;
|
|
|
+
|
|
|
+ QDF_STATUS qdf_status;
|
|
|
+ struct wlan_objmgr_pdev *pdev;
|
|
|
+ struct wlan_objmgr_psoc *psoc;
|
|
|
+ struct wlan_ser_pdev_obj *ser_pdev_obj;
|
|
|
+ struct wlan_serialization_command cmd_bkup;
|
|
|
+ struct wlan_serialization_command_list *cmd_list;
|
|
|
+ struct wlan_serialization_command_list *pcmd_list;
|
|
|
+ struct wlan_serialization_pdev_queue *pdev_queue;
|
|
|
+ bool blocking_cmd_removed = 0;
|
|
|
+ bool blocking_cmd_waiting = 0;
|
|
|
+
|
|
|
+ ser_enter();
|
|
|
+
|
|
|
+ if (!cmd) {
|
|
|
+ ser_err("NULL command");
|
|
|
+ goto error;
|
|
|
+ }
|
|
|
+
|
|
|
+ pdev = wlan_serialization_get_pdev_from_cmd(cmd);
|
|
|
+ if (!pdev) {
|
|
|
+ ser_err("invalid pdev");
|
|
|
+ goto error;
|
|
|
+ }
|
|
|
+
|
|
|
+ psoc = wlan_pdev_get_psoc(pdev);
|
|
|
+ if (!psoc) {
|
|
|
+ ser_err("invalid psoc");
|
|
|
+ goto error;
|
|
|
+ }
|
|
|
+
|
|
|
+ ser_pdev_obj = wlan_serialization_get_pdev_obj(pdev);
|
|
|
+ if (!ser_pdev_obj) {
|
|
|
+ ser_err("ser_pdev_obj is empty");
|
|
|
+ goto error;
|
|
|
+ }
|
|
|
+
|
|
|
+ pdev_queue = wlan_serialization_get_pdev_queue_obj(
|
|
|
+ ser_pdev_obj, cmd->cmd_type);
|
|
|
+
|
|
|
+ ser_debug("dequeue cmd: type[%d] id[%d] high_priority[%d] blocking[%d]",
|
|
|
+ cmd->cmd_type,
|
|
|
+ cmd->cmd_id,
|
|
|
+ cmd->is_high_priority,
|
|
|
+ cmd->is_blocking);
|
|
|
+
|
|
|
+ wlan_serialization_acquire_lock(pdev_queue);
|
|
|
+
|
|
|
+ if (cmd->cmd_type < WLAN_SER_CMD_NONSCAN)
|
|
|
+ qdf_status = wlan_ser_remove_scan_cmd(
|
|
|
+ ser_pdev_obj, &cmd_list, cmd, active_cmd);
|
|
|
+ else {
|
|
|
+ qdf_status = wlan_ser_remove_non_scan_cmd(
|
|
|
+ ser_pdev_obj, &cmd_list, cmd, active_cmd);
|
|
|
+ }
|
|
|
+
|
|
|
+ if (qdf_status != QDF_STATUS_SUCCESS) {
|
|
|
+ wlan_serialization_release_lock(pdev_queue);
|
|
|
+ status = WLAN_SER_CMD_NOT_FOUND;
|
|
|
+ goto error;
|
|
|
+ }
|
|
|
+
|
|
|
+ if (active_cmd) {
|
|
|
+ wlan_serialization_find_and_stop_timer(psoc, &cmd_list->cmd);
|
|
|
+
|
|
|
+ if (cmd_list->cmd.cmd_type >= WLAN_SER_CMD_NONSCAN)
|
|
|
+ blocking_cmd_removed = cmd_list->cmd.is_blocking;
|
|
|
+ }
|
|
|
+
|
|
|
+ qdf_mem_copy(&cmd_bkup, &cmd_list->cmd,
|
|
|
+ sizeof(struct wlan_serialization_command));
|
|
|
+ qdf_mem_zero(&cmd_list->cmd,
|
|
|
+ sizeof(struct wlan_serialization_command));
|
|
|
+ qdf_status = wlan_serialization_insert_back(
|
|
|
+ &pdev_queue->cmd_pool_list,
|
|
|
+ &cmd_list->pdev_node);
|
|
|
+
|
|
|
+ /*
|
|
|
+ * For NON SCAN commands, the following is possible:
|
|
|
+ *
|
|
|
+ * If the remove is for non blocking command,
|
|
|
+ * and there is no blocking command waiting,
|
|
|
+ * look at vdev pending queue and
|
|
|
+ * only one command moves from pending
|
|
|
+ * to active
|
|
|
+ *
|
|
|
+ * If the remove is for blocking comamnd,
|
|
|
+ * look at the pdev queue and
|
|
|
+ * either single blocking command
|
|
|
+ * or multiple non blocking commands moves
|
|
|
+ * from pending to active
|
|
|
+ */
|
|
|
+
|
|
|
+ blocking_cmd_waiting = pdev_queue->blocking_cmd_waiting;
|
|
|
+
|
|
|
+ if (active_cmd) {
|
|
|
+ ser_status = wlan_serialization_move_pending_to_active(
|
|
|
+ cmd_bkup.cmd_type, &pcmd_list, ser_pdev_obj,
|
|
|
+ cmd_bkup.vdev,
|
|
|
+ blocking_cmd_removed,
|
|
|
+ blocking_cmd_waiting);
|
|
|
+ }
|
|
|
+
|
|
|
+ wlan_serialization_release_lock(pdev_queue);
|
|
|
+
|
|
|
+ /* Call cmd cb for remove request*/
|
|
|
+ if (cmd_bkup.cmd_cb) {
|
|
|
+ /* caller should release the memory */
|
|
|
+ ser_debug("cmd cb: type[%d] id[%d]: reason: %s",
|
|
|
+ cmd_bkup.cmd_type,
|
|
|
+ cmd_bkup.cmd_id,
|
|
|
+ "WLAN_SER_CB_RELEASE_MEM_CMD");
|
|
|
+ cmd_bkup.cmd_cb(&cmd_bkup,
|
|
|
+ WLAN_SER_CB_RELEASE_MEM_CMD);
|
|
|
+ }
|
|
|
+
|
|
|
+ /*
|
|
|
+ * If the remove is for non blocking command,
|
|
|
+ * and there is no blocking command waiting,
|
|
|
+ * look at vdev pending queue and
|
|
|
+ * only one command moves from pending
|
|
|
+ * to active and gets activated
|
|
|
+ */
|
|
|
+ if (WLAN_SER_CMD_ACTIVE == ser_status && !blocking_cmd_removed &&
|
|
|
+ !blocking_cmd_waiting) {
|
|
|
+ ser_debug("cmd type[%d] id[%d] moved from pending to active",
|
|
|
+ pcmd_list->cmd.cmd_type,
|
|
|
+ pcmd_list->cmd.cmd_id);
|
|
|
+ wlan_serialization_activate_cmd(pcmd_list,
|
|
|
+ ser_pdev_obj);
|
|
|
+ } else if (ser_status == WLAN_SER_CMD_ACTIVE) {
|
|
|
+ /* If the remove is for blocking command
|
|
|
+ * either one or multiple commands can move
|
|
|
+ * from pending to active and gets activated
|
|
|
+ */
|
|
|
+ wlan_serialization_activate_multiple_cmd(ser_pdev_obj);
|
|
|
+ } else {
|
|
|
+ goto exit;
|
|
|
+ }
|
|
|
+
|
|
|
+exit:
|
|
|
+ if (active_cmd)
|
|
|
+ status = WLAN_SER_CMD_IN_ACTIVE_LIST;
|
|
|
+ else
|
|
|
+ status = WLAN_SER_CMD_IN_PENDING_LIST;
|
|
|
+
|
|
|
+error:
|
|
|
+ ser_exit();
|
|
|
+ return status;
|
|
|
+}
|
|
|
+
|
|
|
+void wlan_serialization_generic_timer_cb(void *arg)
|
|
|
+{
|
|
|
+ struct wlan_serialization_timer *timer = arg;
|
|
|
+ struct wlan_serialization_command *cmd = timer->cmd;
|
|
|
+
|
|
|
+ if (!cmd) {
|
|
|
+ ser_err("command not found");
|
|
|
+ QDF_ASSERT(0);
|
|
|
+ return;
|
|
|
+ }
|
|
|
+
|
|
|
+ ser_err("active cmd timeout for cmd_type[%d] vdev[%pK]",
|
|
|
+ cmd->cmd_type, cmd->vdev);
|
|
|
+
|
|
|
+ if (cmd->cmd_cb)
|
|
|
+ cmd->cmd_cb(cmd, WLAN_SER_CB_ACTIVE_CMD_TIMEOUT);
|
|
|
+
|
|
|
+ /*
|
|
|
+ * dequeue cmd API will cleanup and destroy the timer. If it fails to
|
|
|
+ * dequeue command then we have to destroy the timer.
|
|
|
+ */
|
|
|
+ wlan_serialization_dequeue_cmd(cmd, true);
|
|
|
+}
|
|
|
+
|
|
|
+static QDF_STATUS wlan_serialization_mc_flush_noop(struct scheduler_msg *msg)
|
|
|
+{
|
|
|
+ return QDF_STATUS_SUCCESS;
|
|
|
+}
|
|
|
+
|
|
|
+static void
|
|
|
+wlan_serialization_timer_cb_mc_ctx(void *arg)
|
|
|
+{
|
|
|
+ struct scheduler_msg msg = {0};
|
|
|
+
|
|
|
+ msg.type = SYS_MSG_ID_MC_TIMER;
|
|
|
+ msg.reserved = SYS_MSG_COOKIE;
|
|
|
+ msg.callback = wlan_serialization_generic_timer_cb;
|
|
|
+ msg.bodyptr = arg;
|
|
|
+ msg.bodyval = 0;
|
|
|
+ msg.flush_callback = wlan_serialization_mc_flush_noop;
|
|
|
+
|
|
|
+ if (scheduler_post_msg(QDF_MODULE_ID_SYS, &msg) == QDF_STATUS_SUCCESS)
|
|
|
+ return;
|
|
|
+
|
|
|
+ ser_err("Could not enqueue timer to timer queue");
|
|
|
+}
|
|
|
+
|
|
|
+#ifdef CONFIG_MCL
|
|
|
+static void wlan_serialization_timer_handler(void *arg)
|
|
|
+{
|
|
|
+ ser_enter();
|
|
|
+
|
|
|
+ wlan_serialization_timer_cb_mc_ctx(arg);
|
|
|
+
|
|
|
+ ser_exit();
|
|
|
+}
|
|
|
+#else
|
|
|
+static void wlan_serialization_timer_handler(void *arg)
|
|
|
+{
|
|
|
+ struct wlan_serialization_timer *timer = arg;
|
|
|
+ struct wlan_serialization_command *cmd = timer->cmd;
|
|
|
+
|
|
|
+ if (!cmd) {
|
|
|
+ ser_err("command not found");
|
|
|
+ QDF_ASSERT(0);
|
|
|
+ return;
|
|
|
+ }
|
|
|
+
|
|
|
+ if (cmd->cmd_type < WLAN_SER_CMD_NONSCAN)
|
|
|
+ wlan_serialization_timer_cb_mc_ctx(arg);
|
|
|
+ else
|
|
|
+ wlan_serialization_generic_timer_cb(arg);
|
|
|
+}
|
|
|
+#endif
|
|
|
+
|
|
|
+QDF_STATUS
|
|
|
+wlan_serialization_find_and_stop_timer(struct wlan_objmgr_psoc *psoc,
|
|
|
+ struct wlan_serialization_command *cmd)
|
|
|
+{
|
|
|
+ struct wlan_ser_psoc_obj *psoc_ser_obj;
|
|
|
+ struct wlan_serialization_timer *ser_timer;
|
|
|
+ QDF_STATUS status = QDF_STATUS_E_FAILURE;
|
|
|
+ int i = 0;
|
|
|
+ uint32_t phy_version;
|
|
|
+
|
|
|
+ if (!psoc || !cmd) {
|
|
|
+ ser_err("invalid param");
|
|
|
+ goto error;
|
|
|
+ }
|
|
|
+
|
|
|
+ if (cmd->cmd_timeout_duration == 0) {
|
|
|
+ phy_version = wlan_psoc_get_nif_phy_version(psoc);
|
|
|
+ if (wlan_is_emulation_platform(phy_version)) {
|
|
|
+ ser_err("[SCAN-EMULATION]: Not performing timer funcs");
|
|
|
+ status = QDF_STATUS_SUCCESS;
|
|
|
+ goto exit;
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+ psoc_ser_obj = wlan_serialization_get_psoc_obj(psoc);
|
|
|
+ /*
|
|
|
+ * Here cmd_id and cmd_type are used to locate the timer being
|
|
|
+ * associated with command. For scan command, cmd_id is expected to
|
|
|
+ * be unique and For non-scan command, there should be only one active
|
|
|
+ * command per pdev
|
|
|
+ */
|
|
|
+ for (i = 0; psoc_ser_obj->max_active_cmds > i; i++) {
|
|
|
+ ser_timer = &psoc_ser_obj->timers[i];
|
|
|
+ if (!(ser_timer->cmd) ||
|
|
|
+ (ser_timer->cmd->cmd_id != cmd->cmd_id) ||
|
|
|
+ (ser_timer->cmd->cmd_type != cmd->cmd_type) ||
|
|
|
+ (ser_timer->cmd->vdev != cmd->vdev))
|
|
|
+ continue;
|
|
|
+ status = wlan_serialization_stop_timer(ser_timer);
|
|
|
+ ser_debug("\n Stopping timer for cmd type:%d, id: %d",
|
|
|
+ cmd->cmd_type, cmd->cmd_id);
|
|
|
+ break;
|
|
|
+ }
|
|
|
+
|
|
|
+ if (QDF_STATUS_SUCCESS != status)
|
|
|
+ ser_err("Can't find timer for cmd_type[%d]", cmd->cmd_type);
|
|
|
+
|
|
|
+error:
|
|
|
+exit:
|
|
|
+
|
|
|
+ return status;
|
|
|
+}
|
|
|
+
|
|
|
+QDF_STATUS
|
|
|
+wlan_serialization_find_and_start_timer(struct wlan_objmgr_psoc *psoc,
|
|
|
+ struct wlan_serialization_command *cmd)
|
|
|
+{
|
|
|
+ QDF_STATUS status = QDF_STATUS_E_FAILURE;
|
|
|
+ struct wlan_ser_psoc_obj *psoc_ser_obj;
|
|
|
+ struct wlan_serialization_timer *ser_timer;
|
|
|
+ int i = 0;
|
|
|
+ uint32_t nif_phy_ver;
|
|
|
+
|
|
|
+ if (!psoc || !cmd) {
|
|
|
+ ser_err("invalid param");
|
|
|
+ goto error;
|
|
|
+ }
|
|
|
+
|
|
|
+ nif_phy_ver = wlan_psoc_get_nif_phy_version(psoc);
|
|
|
+ if ((cmd->cmd_timeout_duration == 0) &&
|
|
|
+ (wlan_is_emulation_platform(nif_phy_ver))) {
|
|
|
+ ser_err("[SCAN-EMULATION]: Not performing timer functions\n");
|
|
|
+ status = QDF_STATUS_SUCCESS;
|
|
|
+ goto exit;
|
|
|
+ }
|
|
|
+
|
|
|
+ psoc_ser_obj = wlan_serialization_get_psoc_obj(psoc);
|
|
|
+
|
|
|
+ for (i = 0; psoc_ser_obj->max_active_cmds > i; i++) {
|
|
|
+ /* Keep trying timer */
|
|
|
+ ser_timer = &psoc_ser_obj->timers[i];
|
|
|
+ if (ser_timer->cmd)
|
|
|
+ continue;
|
|
|
+
|
|
|
+ /* Remember timer is pointing to command */
|
|
|
+ ser_timer->cmd = cmd;
|
|
|
+ qdf_timer_init(NULL,
|
|
|
+ &ser_timer->timer,
|
|
|
+ wlan_serialization_timer_handler,
|
|
|
+ ser_timer,
|
|
|
+ QDF_TIMER_TYPE_SW);
|
|
|
+ qdf_timer_mod(&ser_timer->timer,
|
|
|
+ cmd->cmd_timeout_duration);
|
|
|
+
|
|
|
+ ser_debug("starting timer for cmd: type[%d] id[%d] high_priority[%d] blocking[%d]",
|
|
|
+ cmd->cmd_type,
|
|
|
+ cmd->cmd_id,
|
|
|
+ cmd->is_high_priority,
|
|
|
+ cmd->is_blocking);
|
|
|
+
|
|
|
+ status = QDF_STATUS_SUCCESS;
|
|
|
+ break;
|
|
|
+ }
|
|
|
+
|
|
|
+error:
|
|
|
+exit:
|
|
|
+
|
|
|
+ return status;
|
|
|
+}
|
|
|
+
|
|
|
+enum wlan_serialization_cmd_status
|
|
|
+wlan_serialization_cmd_cancel_handler(
|
|
|
+ struct wlan_ser_pdev_obj *ser_obj,
|
|
|
+ struct wlan_serialization_command *cmd,
|
|
|
+ struct wlan_objmgr_pdev *pdev, struct wlan_objmgr_vdev *vdev,
|
|
|
+ enum wlan_serialization_cmd_type cmd_type, uint8_t queue_type)
|
|
|
+{
|
|
|
+ enum wlan_serialization_cmd_status active_status =
|
|
|
+ WLAN_SER_CMD_NOT_FOUND;
|
|
|
+ enum wlan_serialization_cmd_status pending_status =
|
|
|
+ WLAN_SER_CMD_NOT_FOUND;
|
|
|
+ enum wlan_serialization_cmd_status status =
|
|
|
+ WLAN_SER_CMD_NOT_FOUND;
|
|
|
+
|
|
|
+ ser_enter();
|
|
|
+
|
|
|
+ if (!ser_obj) {
|
|
|
+ ser_err("invalid serial object");
|
|
|
+ goto error;
|
|
|
+ }
|
|
|
+
|
|
|
+ if (queue_type & WLAN_SERIALIZATION_ACTIVE_QUEUE) {
|
|
|
+ if (cmd_type < WLAN_SER_CMD_NONSCAN)
|
|
|
+ active_status = wlan_ser_cancel_scan_cmd(
|
|
|
+ ser_obj, pdev, vdev, cmd,
|
|
|
+ cmd_type, true);
|
|
|
+ else
|
|
|
+ active_status = wlan_ser_cancel_non_scan_cmd(
|
|
|
+ ser_obj, pdev, vdev, cmd,
|
|
|
+ cmd_type, true);
|
|
|
+ }
|
|
|
+
|
|
|
+ if (queue_type & WLAN_SERIALIZATION_PENDING_QUEUE) {
|
|
|
+ if (cmd_type < WLAN_SER_CMD_NONSCAN)
|
|
|
+ pending_status = wlan_ser_cancel_scan_cmd(
|
|
|
+ ser_obj, pdev, vdev, cmd,
|
|
|
+ cmd_type, false);
|
|
|
+ else
|
|
|
+ pending_status = wlan_ser_cancel_non_scan_cmd(
|
|
|
+ ser_obj, pdev, vdev, cmd,
|
|
|
+ cmd_type, false);
|
|
|
+ }
|
|
|
+
|
|
|
+ if (active_status == WLAN_SER_CMD_IN_ACTIVE_LIST &&
|
|
|
+ pending_status == WLAN_SER_CMD_IN_PENDING_LIST)
|
|
|
+ status = WLAN_SER_CMDS_IN_ALL_LISTS;
|
|
|
+ else if (active_status == WLAN_SER_CMD_IN_ACTIVE_LIST)
|
|
|
+ status = active_status;
|
|
|
+ else if (pending_status == WLAN_SER_CMD_IN_PENDING_LIST)
|
|
|
+ status = pending_status;
|
|
|
+
|
|
|
+error:
|
|
|
+ ser_exit();
|
|
|
+ return status;
|
|
|
+}
|
|
|
+
|
|
|
+enum wlan_serialization_cmd_status
|
|
|
+wlan_serialization_find_and_cancel_cmd(
|
|
|
+ struct wlan_serialization_command *cmd,
|
|
|
+ enum wlan_serialization_cancel_type req_type,
|
|
|
+ uint8_t queue_type)
|
|
|
+{
|
|
|
+ enum wlan_serialization_cmd_status status = WLAN_SER_CMD_NOT_FOUND;
|
|
|
+ struct wlan_ser_pdev_obj *ser_obj = NULL;
|
|
|
+ struct wlan_objmgr_pdev *pdev;
|
|
|
+
|
|
|
+ ser_enter();
|
|
|
+
|
|
|
+ if (!cmd) {
|
|
|
+ ser_err("Invalid cmd");
|
|
|
+ goto error;
|
|
|
+ }
|
|
|
+
|
|
|
+ pdev = wlan_serialization_get_pdev_from_cmd(cmd);
|
|
|
+ if (!pdev) {
|
|
|
+ ser_err("Invalid pdev");
|
|
|
+ goto error;
|
|
|
+ }
|
|
|
+ ser_obj = wlan_serialization_get_pdev_obj(pdev);
|
|
|
+ if (!ser_obj) {
|
|
|
+ ser_err("Invalid ser_obj");
|
|
|
+ goto error;
|
|
|
+ }
|
|
|
+
|
|
|
+ switch (req_type) {
|
|
|
+ case WLAN_SER_CANCEL_SINGLE_SCAN:
|
|
|
+ /* remove scan cmd which matches the given cmd struct */
|
|
|
+ status = wlan_serialization_cmd_cancel_handler(ser_obj,
|
|
|
+ cmd,
|
|
|
+ NULL,
|
|
|
+ NULL,
|
|
|
+ cmd->cmd_type,
|
|
|
+ queue_type);
|
|
|
+ break;
|
|
|
+ case WLAN_SER_CANCEL_PDEV_SCANS:
|
|
|
+ /* remove all scan cmds which matches the pdev object */
|
|
|
+ status = wlan_serialization_cmd_cancel_handler(
|
|
|
+ ser_obj,
|
|
|
+ NULL,
|
|
|
+ wlan_vdev_get_pdev(cmd->vdev),
|
|
|
+ NULL,
|
|
|
+ cmd->cmd_type,
|
|
|
+ queue_type);
|
|
|
+ break;
|
|
|
+ case WLAN_SER_CANCEL_VDEV_SCANS:
|
|
|
+ /* remove all scan cmds which matches the vdev object */
|
|
|
+ status = wlan_serialization_cmd_cancel_handler(ser_obj,
|
|
|
+ NULL, NULL,
|
|
|
+ cmd->vdev,
|
|
|
+ cmd->cmd_type,
|
|
|
+ queue_type);
|
|
|
+ break;
|
|
|
+ case WLAN_SER_CANCEL_NON_SCAN_CMD:
|
|
|
+ /* remove nonscan cmd which matches the given cmd */
|
|
|
+ status = wlan_serialization_cmd_cancel_handler(ser_obj,
|
|
|
+ cmd,
|
|
|
+ NULL,
|
|
|
+ NULL,
|
|
|
+ cmd->cmd_type,
|
|
|
+ queue_type);
|
|
|
+ break;
|
|
|
+ case WLAN_SER_CANCEL_PDEV_NON_SCAN_CMD:
|
|
|
+ /* remove all non scan cmds which matches the pdev object */
|
|
|
+ status = wlan_serialization_cmd_cancel_handler(
|
|
|
+ ser_obj,
|
|
|
+ NULL,
|
|
|
+ wlan_vdev_get_pdev(cmd->vdev),
|
|
|
+ NULL,
|
|
|
+ cmd->cmd_type,
|
|
|
+ queue_type);
|
|
|
+ break;
|
|
|
+ case WLAN_SER_CANCEL_VDEV_NON_SCAN_CMD:
|
|
|
+ /* remove all non scan cmds which matches the vdev object */
|
|
|
+ status = wlan_serialization_cmd_cancel_handler(ser_obj,
|
|
|
+ NULL, NULL,
|
|
|
+ cmd->vdev,
|
|
|
+ cmd->cmd_type,
|
|
|
+ queue_type);
|
|
|
+ break;
|
|
|
+ default:
|
|
|
+ ser_err("Invalid request");
|
|
|
+ }
|
|
|
+
|
|
|
+error:
|
|
|
+ ser_exit();
|
|
|
+ return status;
|
|
|
+}
|