summaryrefslogtreecommitdiff
path: root/qdf/linux/src/qdf_mc_timer.c
diff options
context:
space:
mode:
Diffstat (limited to 'qdf/linux/src/qdf_mc_timer.c')
-rw-r--r--qdf/linux/src/qdf_mc_timer.c702
1 files changed, 702 insertions, 0 deletions
diff --git a/qdf/linux/src/qdf_mc_timer.c b/qdf/linux/src/qdf_mc_timer.c
new file mode 100644
index 000000000000..858c325f4412
--- /dev/null
+++ b/qdf/linux/src/qdf_mc_timer.c
@@ -0,0 +1,702 @@
+/*
+ * Copyright (c) 2014-2016 The Linux Foundation. All rights reserved.
+ *
+ * Previously licensed under the ISC license by Qualcomm Atheros, Inc.
+ *
+ *
+ * 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.
+ */
+
+/*
+ * This file was originally distributed by Qualcomm Atheros, Inc.
+ * under proprietary terms before Copyright ownership was assigned
+ * to the Linux Foundation.
+ */
+
+/**
+ * DOC: qdf_mc_timer
+ * QCA driver framework timer APIs serialized to MC thread
+ */
+
+/* Include Files */
+#include <qdf_mc_timer.h>
+#include <qdf_lock.h>
+#include "qdf_lock.h"
+#include "qdf_list.h"
+#include "qdf_mem.h"
+#ifdef CONFIG_MCL
+#include <cds_mc_timer.h>
+#endif
+/* Preprocessor definitions and constants */
+
+#define LINUX_TIMER_COOKIE 0x12341234
+#define LINUX_INVALID_TIMER_COOKIE 0xfeedface
+#define TMR_INVALID_ID (0)
+
+/* Type declarations */
+
+/* Static Variable Definitions */
+static unsigned int persistent_timer_count;
+static qdf_mutex_t persistent_timer_count_lock;
+
+/* Function declarations and documenation */
+
+/**
+ * qdf_try_allowing_sleep() - clean up timer states after it has been deactivated
+ * @type: timer type
+ *
+ * Clean up timer states after it has been deactivated check and try to allow
+ * sleep after a timer has been stopped or expired.
+ *
+ * Return: none
+ */
+void qdf_try_allowing_sleep(QDF_TIMER_TYPE type)
+{
+ if (QDF_TIMER_TYPE_WAKE_APPS == type) {
+
+ persistent_timer_count--;
+ if (0 == persistent_timer_count) {
+ /* since the number of persistent timers has
+ decreased from 1 to 0, the timer should allow
+ sleep
+ */
+ }
+ }
+}
+EXPORT_SYMBOL(qdf_try_allowing_sleep);
+
+/**
+ * qdf_mc_timer_get_current_state() - get the current state of the timer
+ * @timer: Pointer to timer object
+ *
+ * Return:
+ * QDF_TIMER_STATE - qdf timer state
+ */
+QDF_TIMER_STATE qdf_mc_timer_get_current_state(qdf_mc_timer_t *timer)
+{
+ if (NULL == timer) {
+ QDF_ASSERT(0);
+ return QDF_TIMER_STATE_UNUSED;
+ }
+
+ switch (timer->state) {
+ case QDF_TIMER_STATE_STOPPED:
+ case QDF_TIMER_STATE_STARTING:
+ case QDF_TIMER_STATE_RUNNING:
+ case QDF_TIMER_STATE_UNUSED:
+ return timer->state;
+ default:
+ QDF_ASSERT(0);
+ return QDF_TIMER_STATE_UNUSED;
+ }
+}
+EXPORT_SYMBOL(qdf_mc_timer_get_current_state);
+
+/**
+ * qdf_timer_module_init() - initializes a QDF timer module.
+ *
+ * This API initializes the QDF timer module. This needs to be called
+ * exactly once prior to using any QDF timers.
+ *
+ * Return: none
+ */
+void qdf_timer_module_init(void)
+{
+ QDF_TRACE(QDF_MODULE_ID_QDF, QDF_TRACE_LEVEL_INFO_HIGH,
+ "Initializing the QDF MC timer module");
+ qdf_mutex_create(&persistent_timer_count_lock);
+}
+EXPORT_SYMBOL(qdf_timer_module_init);
+
+#ifdef TIMER_MANAGER
+
+qdf_list_t qdf_timer_list;
+qdf_spinlock_t qdf_timer_list_lock;
+
+static void qdf_timer_clean(void);
+
+/**
+ * qdf_mc_timer_manager_init() - initialize QDF debug timer manager
+ *
+ * This API initializes QDF timer debug functionality.
+ *
+ * Return: none
+ */
+void qdf_mc_timer_manager_init(void)
+{
+ qdf_list_create(&qdf_timer_list, 1000);
+ qdf_spinlock_create(&qdf_timer_list_lock);
+ return;
+}
+EXPORT_SYMBOL(qdf_mc_timer_manager_init);
+
+/**
+ * qdf_timer_clean() - clean up QDF timer debug functionality
+ *
+ * This API cleans up QDF timer debug functionality and prints which QDF timers
+ * are leaked. This is called during driver unload.
+ *
+ * Return: none
+ */
+static void qdf_timer_clean(void)
+{
+ uint32_t list_size;
+ qdf_list_node_t *node;
+ QDF_STATUS qdf_status;
+
+ qdf_mc_timer_node_t *timer_node;
+
+ list_size = qdf_list_size(&qdf_timer_list);
+
+ if (!list_size)
+ return;
+ QDF_TRACE(QDF_MODULE_ID_QDF, QDF_TRACE_LEVEL_INFO_HIGH,
+ "%s: List is not Empty. list_size %d ",
+ __func__, (int)list_size);
+
+ do {
+ qdf_spin_lock_irqsave(&qdf_timer_list_lock);
+ qdf_status = qdf_list_remove_front(&qdf_timer_list, &node);
+ qdf_spin_unlock_irqrestore(&qdf_timer_list_lock);
+ if (QDF_STATUS_SUCCESS == qdf_status) {
+ timer_node = (qdf_mc_timer_node_t *) node;
+ QDF_TRACE(QDF_MODULE_ID_QDF, QDF_TRACE_LEVEL_FATAL,
+ "timer Leak@ File %s, @Line %d",
+ timer_node->file_name,
+ (int)timer_node->line_num);
+ qdf_mem_free(timer_node);
+ }
+ } while (qdf_status == QDF_STATUS_SUCCESS);
+}
+EXPORT_SYMBOL(qdf_timer_clean);
+
+/**
+ * qdf_mc_timer_manager_exit() - exit QDF timer debug functionality
+ *
+ * This API exists QDF timer debug functionality
+ *
+ * Return: none
+ */
+void qdf_mc_timer_manager_exit(void)
+{
+ qdf_timer_clean();
+ qdf_list_destroy(&qdf_timer_list);
+}
+EXPORT_SYMBOL(qdf_mc_timer_manager_exit);
+#endif
+
+/**
+ * qdf_mc_timer_init() - initialize a QDF timer
+ * @timer: Pointer to timer object
+ * @timer_type: Type of timer
+ * @callback: Callback to be called after timer expiry
+ * @ser_data: User data which will be passed to callback function
+ *
+ * This API initializes a QDF timer object.
+ *
+ * qdf_mc_timer_init() initializes a QDF timer object. A timer must be
+ * initialized by calling qdf_mc_timer_initialize() before it may be used in
+ * any other timer functions.
+ *
+ * Attempting to initialize timer that is already initialized results in
+ * a failure. A destroyed timer object can be re-initialized with a call to
+ * qdf_mc_timer_init(). The results of otherwise referencing the object
+ * after it has been destroyed are undefined.
+ *
+ * Calls to QDF timer functions to manipulate the timer such
+ * as qdf_mc_timer_set() will fail if the timer is not initialized or has
+ * been destroyed. Therefore, don't use the timer after it has been
+ * destroyed until it has been re-initialized.
+ *
+ * All callback will be executed within the CDS main thread unless it is
+ * initialized from the Tx thread flow, in which case it will be executed
+ * within the tx thread flow.
+ *
+ * Return:
+ * QDF_STATUS_SUCCESS: timer is initialized successfully
+ * QDF failure status: timer initialization failed
+ */
+#ifdef TIMER_MANAGER
+QDF_STATUS qdf_mc_timer_init_debug(qdf_mc_timer_t *timer,
+ QDF_TIMER_TYPE timer_type,
+ qdf_mc_timer_callback_t callback,
+ void *user_data, char *file_name,
+ uint32_t line_num)
+{
+ QDF_STATUS qdf_status;
+
+ /* check for invalid pointer */
+ if ((timer == NULL) || (callback == NULL)) {
+ QDF_TRACE(QDF_MODULE_ID_QDF, QDF_TRACE_LEVEL_ERROR,
+ "%s: Null params being passed", __func__);
+ QDF_ASSERT(0);
+ return QDF_STATUS_E_FAULT;
+ }
+
+ timer->timer_node = qdf_mem_malloc(sizeof(qdf_mc_timer_node_t));
+
+ if (timer->timer_node == NULL) {
+ QDF_TRACE(QDF_MODULE_ID_QDF, QDF_TRACE_LEVEL_ERROR,
+ "%s: Not able to allocate memory for time_node",
+ __func__);
+ QDF_ASSERT(0);
+ return QDF_STATUS_E_NOMEM;
+ }
+
+ qdf_mem_set(timer->timer_node, sizeof(qdf_mc_timer_node_t), 0);
+
+ timer->timer_node->file_name = file_name;
+ timer->timer_node->line_num = line_num;
+ timer->timer_node->qdf_timer = timer;
+
+ qdf_spin_lock_irqsave(&qdf_timer_list_lock);
+ qdf_status = qdf_list_insert_front(&qdf_timer_list,
+ &timer->timer_node->node);
+ qdf_spin_unlock_irqrestore(&qdf_timer_list_lock);
+ if (QDF_STATUS_SUCCESS != qdf_status) {
+ QDF_TRACE(QDF_MODULE_ID_QDF, QDF_TRACE_LEVEL_ERROR,
+ "%s: Unable to insert node into List qdf_status %d",
+ __func__, qdf_status);
+ }
+
+ /* set the various members of the timer structure
+ * with arguments passed or with default values
+ */
+ spin_lock_init(&timer->platform_info.spinlock);
+ if (QDF_TIMER_TYPE_SW == timer_type)
+ init_timer_deferrable(&(timer->platform_info.timer));
+ else
+ init_timer(&(timer->platform_info.timer));
+#ifdef CONFIG_MCL
+ timer->platform_info.timer.function = cds_linux_timer_callback;
+#else
+ timer->platform_info.timer.function = NULL;
+#endif
+ timer->platform_info.timer.data = (unsigned long)timer;
+ timer->callback = callback;
+ timer->user_data = user_data;
+ timer->type = timer_type;
+ timer->platform_info.cookie = LINUX_TIMER_COOKIE;
+ timer->platform_info.thread_id = 0;
+ timer->state = QDF_TIMER_STATE_STOPPED;
+
+ return QDF_STATUS_SUCCESS;
+}
+#else
+QDF_STATUS qdf_mc_timer_init(qdf_mc_timer_t *timer, QDF_TIMER_TYPE timer_type,
+ qdf_mc_timer_callback_t callback,
+ void *user_data)
+{
+ /* check for invalid pointer */
+ if ((timer == NULL) || (callback == NULL)) {
+ QDF_TRACE(QDF_MODULE_ID_QDF, QDF_TRACE_LEVEL_ERROR,
+ "%s: Null params being passed", __func__);
+ QDF_ASSERT(0);
+ return QDF_STATUS_E_FAULT;
+ }
+
+ /* set the various members of the timer structure
+ * with arguments passed or with default values
+ */
+ spin_lock_init(&timer->platform_info.spinlock);
+ if (QDF_TIMER_TYPE_SW == timer_type)
+ init_timer_deferrable(&(timer->platform_info.timer));
+ else
+ init_timer(&(timer->platform_info.timer));
+#ifdef CONFIG_MCL
+ timer->platform_info.timer.function = cds_linux_timer_callback;
+#else
+ timer->platform_info.timer.function = NULL;
+#endif
+ timer->platform_info.timer.data = (unsigned long)timer;
+ timer->callback = callback;
+ timer->user_data = user_data;
+ timer->type = timer_type;
+ timer->platform_info.cookie = LINUX_TIMER_COOKIE;
+ timer->platform_info.thread_id = 0;
+ timer->state = QDF_TIMER_STATE_STOPPED;
+
+ return QDF_STATUS_SUCCESS;
+}
+#endif
+
+/**
+ * qdf_mc_timer_destroy() - destroy QDF timer
+ * @timer: Pointer to timer object
+ *
+ * qdf_mc_timer_destroy() function shall destroy the timer object.
+ * After a successful return from \a qdf_mc_timer_destroy() the timer
+ * object becomes, in effect, uninitialized.
+ *
+ * A destroyed timer object can be re-initialized by calling
+ * qdf_mc_timer_init(). The results of otherwise referencing the object
+ * after it has been destroyed are undefined.
+ *
+ * Calls to QDF timer functions to manipulate the timer, such
+ * as qdf_mc_timer_set() will fail if the lock is destroyed. Therefore,
+ * don't use the timer after it has been destroyed until it has
+ * been re-initialized.
+ *
+ * Return:
+ * QDF_STATUS_SUCCESS - timer is initialized successfully
+ * QDF failure status - timer initialization failed
+ */
+#ifdef TIMER_MANAGER
+QDF_STATUS qdf_mc_timer_destroy(qdf_mc_timer_t *timer)
+{
+ QDF_STATUS v_status = QDF_STATUS_SUCCESS;
+ unsigned long flags;
+
+ /* check for invalid pointer */
+ if (NULL == timer) {
+ QDF_TRACE(QDF_MODULE_ID_QDF, QDF_TRACE_LEVEL_ERROR,
+ "%s: Null timer pointer being passed", __func__);
+ QDF_ASSERT(0);
+ return QDF_STATUS_E_FAULT;
+ }
+
+ /* Check if timer refers to an uninitialized object */
+ if (LINUX_TIMER_COOKIE != timer->platform_info.cookie) {
+ QDF_TRACE(QDF_MODULE_ID_QDF, QDF_TRACE_LEVEL_ERROR,
+ "%s: Cannot destroy uninitialized timer", __func__);
+ QDF_ASSERT(0);
+ return QDF_STATUS_E_INVAL;
+ }
+
+ qdf_spin_lock_irqsave(&qdf_timer_list_lock);
+ v_status = qdf_list_remove_node(&qdf_timer_list,
+ &timer->timer_node->node);
+ qdf_spin_unlock_irqrestore(&qdf_timer_list_lock);
+ if (v_status != QDF_STATUS_SUCCESS) {
+ QDF_ASSERT(0);
+ return QDF_STATUS_E_INVAL;
+ }
+ qdf_mem_free(timer->timer_node);
+
+ spin_lock_irqsave(&timer->platform_info.spinlock, flags);
+
+ switch (timer->state) {
+
+ case QDF_TIMER_STATE_STARTING:
+ v_status = QDF_STATUS_E_BUSY;
+ break;
+
+ case QDF_TIMER_STATE_RUNNING:
+ /* Stop the timer first */
+ del_timer(&(timer->platform_info.timer));
+ v_status = QDF_STATUS_SUCCESS;
+ break;
+ case QDF_TIMER_STATE_STOPPED:
+ v_status = QDF_STATUS_SUCCESS;
+ break;
+
+ case QDF_TIMER_STATE_UNUSED:
+ v_status = QDF_STATUS_E_ALREADY;
+ break;
+
+ default:
+ v_status = QDF_STATUS_E_FAULT;
+ break;
+ }
+
+ if (QDF_STATUS_SUCCESS == v_status) {
+ timer->platform_info.cookie = LINUX_INVALID_TIMER_COOKIE;
+ timer->state = QDF_TIMER_STATE_UNUSED;
+ spin_unlock_irqrestore(&timer->platform_info.spinlock, flags);
+ return v_status;
+ }
+
+ spin_unlock_irqrestore(&timer->platform_info.spinlock, flags);
+
+ QDF_TRACE(QDF_MODULE_ID_QDF, QDF_TRACE_LEVEL_INFO_HIGH,
+ "%s: Cannot destroy timer in state = %d", __func__,
+ timer->state);
+ QDF_ASSERT(0);
+
+ return v_status;
+}
+EXPORT_SYMBOL(qdf_mc_timer_destroy);
+
+#else
+
+/**
+ * qdf_mc_timer_destroy() - destroy QDF timer
+ * @timer: Pointer to timer object
+ *
+ * qdf_mc_timer_destroy() function shall destroy the timer object.
+ * After a successful return from \a qdf_mc_timer_destroy() the timer
+ * object becomes, in effect, uninitialized.
+ *
+ * A destroyed timer object can be re-initialized by calling
+ * qdf_mc_timer_init(). The results of otherwise referencing the object
+ * after it has been destroyed are undefined.
+ *
+ * Calls to QDF timer functions to manipulate the timer, such
+ * as qdf_mc_timer_set() will fail if the lock is destroyed. Therefore,
+ * don't use the timer after it has been destroyed until it has
+ * been re-initialized.
+ *
+ * Return:
+ * QDF_STATUS_SUCCESS - timer is initialized successfully
+ * QDF failure status - timer initialization failed
+ */
+QDF_STATUS qdf_mc_timer_destroy(qdf_mc_timer_t *timer)
+{
+ QDF_STATUS v_status = QDF_STATUS_SUCCESS;
+ unsigned long flags;
+
+ /* check for invalid pointer */
+ if (NULL == timer) {
+ QDF_TRACE(QDF_MODULE_ID_QDF, QDF_TRACE_LEVEL_ERROR,
+ "%s: Null timer pointer being passed", __func__);
+ QDF_ASSERT(0);
+ return QDF_STATUS_E_FAULT;
+ }
+
+ /* check if timer refers to an uninitialized object */
+ if (LINUX_TIMER_COOKIE != timer->platform_info.cookie) {
+ QDF_TRACE(QDF_MODULE_ID_QDF, QDF_TRACE_LEVEL_ERROR,
+ "%s: Cannot destroy uninitialized timer", __func__);
+ QDF_ASSERT(0);
+ return QDF_STATUS_E_INVAL;
+ }
+ spin_lock_irqsave(&timer->platform_info.spinlock, flags);
+
+ switch (timer->state) {
+
+ case QDF_TIMER_STATE_STARTING:
+ v_status = QDF_STATUS_E_BUSY;
+ break;
+
+ case QDF_TIMER_STATE_RUNNING:
+ /* Stop the timer first */
+ del_timer(&(timer->platform_info.timer));
+ v_status = QDF_STATUS_SUCCESS;
+ break;
+
+ case QDF_TIMER_STATE_STOPPED:
+ v_status = QDF_STATUS_SUCCESS;
+ break;
+
+ case QDF_TIMER_STATE_UNUSED:
+ v_status = QDF_STATUS_E_ALREADY;
+ break;
+
+ default:
+ v_status = QDF_STATUS_E_FAULT;
+ break;
+ }
+
+ if (QDF_STATUS_SUCCESS == v_status) {
+ timer->platform_info.cookie = LINUX_INVALID_TIMER_COOKIE;
+ timer->state = QDF_TIMER_STATE_UNUSED;
+ spin_unlock_irqrestore(&timer->platform_info.spinlock, flags);
+ return v_status;
+ }
+
+ spin_unlock_irqrestore(&timer->platform_info.spinlock, flags);
+
+ QDF_TRACE(QDF_MODULE_ID_QDF, QDF_TRACE_LEVEL_INFO_HIGH,
+ "%s: Cannot destroy timer in state = %d", __func__,
+ timer->state);
+ QDF_ASSERT(0);
+
+ return v_status;
+}
+EXPORT_SYMBOL(qdf_mc_timer_destroy);
+#endif
+
+/**
+ * qdf_mc_timer_start() - start a QDF timer object
+ * @timer: Pointer to timer object
+ * @expiration_time: Time to expire
+ *
+ * qdf_mc_timer_start() function starts a timer to expire after the
+ * specified interval, thus running the timer callback function when
+ * the interval expires.
+ *
+ * A timer only runs once (a one-shot timer). To re-start the
+ * timer, qdf_mc_timer_start() has to be called after the timer runs
+ * or has been cancelled.
+ *
+ * Return:
+ * QDF_STATUS_SUCCESS: timer is initialized successfully
+ * QDF failure status: timer initialization failed
+ */
+QDF_STATUS qdf_mc_timer_start(qdf_mc_timer_t *timer, uint32_t expiration_time)
+{
+ unsigned long flags;
+
+ QDF_TRACE(QDF_MODULE_ID_QDF, QDF_TRACE_LEVEL_INFO_HIGH,
+ "timer Addr inside qdf_mc_timer_start : 0x%p ", timer);
+
+ /* check for invalid pointer */
+ if (NULL == timer) {
+ QDF_TRACE(QDF_MODULE_ID_QDF, QDF_TRACE_LEVEL_ERROR,
+ "%s Null timer pointer being passed", __func__);
+ QDF_ASSERT(0);
+ return QDF_STATUS_E_INVAL;
+ }
+
+ /* check if timer refers to an uninitialized object */
+ if (LINUX_TIMER_COOKIE != timer->platform_info.cookie) {
+ QDF_TRACE(QDF_MODULE_ID_QDF, QDF_TRACE_LEVEL_ERROR,
+ "%s: Cannot start uninitialized timer", __func__);
+ QDF_ASSERT(0);
+
+ return QDF_STATUS_E_INVAL;
+ }
+
+ /* check if timer has expiration time less than 10 ms */
+ if (expiration_time < 10) {
+ QDF_TRACE(QDF_MODULE_ID_QDF, QDF_TRACE_LEVEL_ERROR,
+ "%s: Cannot start a timer with expiration less than 10 ms",
+ __func__);
+ QDF_ASSERT(0);
+ return QDF_STATUS_E_INVAL;
+ }
+
+ /* make sure the remainer of the logic isn't interrupted */
+ spin_lock_irqsave(&timer->platform_info.spinlock, flags);
+
+ /* ensure if the timer can be started */
+ if (QDF_TIMER_STATE_STOPPED != timer->state) {
+ spin_unlock_irqrestore(&timer->platform_info.spinlock, flags);
+ QDF_TRACE(QDF_MODULE_ID_QDF, QDF_TRACE_LEVEL_ERROR,
+ "%s: Cannot start timer in state = %d ", __func__,
+ timer->state);
+ return QDF_STATUS_E_ALREADY;
+ }
+
+ /* start the timer */
+ mod_timer(&(timer->platform_info.timer),
+ jiffies + msecs_to_jiffies(expiration_time));
+
+ timer->state = QDF_TIMER_STATE_RUNNING;
+
+ /* get the thread ID on which the timer is being started */
+ timer->platform_info.thread_id = current->pid;
+
+ if (QDF_TIMER_TYPE_WAKE_APPS == timer->type) {
+ persistent_timer_count++;
+ if (1 == persistent_timer_count) {
+ /* since we now have one persistent timer,
+ * we need to disallow sleep
+ * sleep_negate_okts(sleep_client_handle);
+ */
+ }
+ }
+
+ spin_unlock_irqrestore(&timer->platform_info.spinlock, flags);
+
+ return QDF_STATUS_SUCCESS;
+}
+EXPORT_SYMBOL(qdf_mc_timer_start);
+
+/**
+ * qdf_mc_timer_stop() - stop a QDF timer
+ * @timer: Pointer to timer object
+ * qdf_mc_timer_stop() function stops a timer that has been started but
+ * has not expired, essentially cancelling the 'start' request.
+ *
+ * After a timer is stopped, it goes back to the state it was in after it
+ * was created and can be started again via a call to qdf_mc_timer_start().
+ *
+ * Return:
+ * QDF_STATUS_SUCCESS: timer is initialized successfully
+ * QDF failure status: timer initialization failed
+ */
+QDF_STATUS qdf_mc_timer_stop(qdf_mc_timer_t *timer)
+{
+ unsigned long flags;
+
+ QDF_TRACE(QDF_MODULE_ID_QDF, QDF_TRACE_LEVEL_INFO_HIGH,
+ "%s: timer Addr inside qdf_mc_timer_stop : 0x%p",
+ __func__, timer);
+
+ /* check for invalid pointer */
+ if (NULL == timer) {
+ QDF_TRACE(QDF_MODULE_ID_QDF, QDF_TRACE_LEVEL_ERROR,
+ "%s Null timer pointer being passed", __func__);
+ QDF_ASSERT(0);
+ return QDF_STATUS_E_INVAL;
+ }
+
+ /* check if timer refers to an uninitialized object */
+ if (LINUX_TIMER_COOKIE != timer->platform_info.cookie) {
+ QDF_TRACE(QDF_MODULE_ID_QDF, QDF_TRACE_LEVEL_ERROR,
+ "%s: Cannot stop uninitialized timer", __func__);
+ QDF_ASSERT(0);
+
+ return QDF_STATUS_E_INVAL;
+ }
+
+ /* ensure the timer state is correct */
+ spin_lock_irqsave(&timer->platform_info.spinlock, flags);
+
+ if (QDF_TIMER_STATE_RUNNING != timer->state) {
+ spin_unlock_irqrestore(&timer->platform_info.spinlock, flags);
+ QDF_TRACE(QDF_MODULE_ID_QDF, QDF_TRACE_LEVEL_INFO_HIGH,
+ "%s: Cannot stop timer in state = %d",
+ __func__, timer->state);
+ return QDF_STATUS_SUCCESS;
+ }
+
+ timer->state = QDF_TIMER_STATE_STOPPED;
+
+ del_timer(&(timer->platform_info.timer));
+
+ spin_unlock_irqrestore(&timer->platform_info.spinlock, flags);
+
+ qdf_try_allowing_sleep(timer->type);
+
+ return QDF_STATUS_SUCCESS;
+}
+EXPORT_SYMBOL(qdf_mc_timer_stop);
+
+/**
+ * qdf_mc_timer_get_system_ticks() - get the system time in 10ms ticks
+
+ * qdf_mc_timer_get_system_ticks() function returns the current number
+ * of timer ticks in 10msec intervals. This function is suitable timestamping
+ * and calculating time intervals by calculating the difference between two
+ * timestamps.
+ *
+ * Return:
+ * The current system tick count (in 10msec intervals). This
+ * function cannot fail.
+ */
+unsigned long qdf_mc_timer_get_system_ticks(void)
+{
+ return jiffies_to_msecs(jiffies) / 10;
+}
+EXPORT_SYMBOL(qdf_mc_timer_get_system_ticks);
+
+/**
+ * qdf_mc_timer_get_system_time() - Get the system time in milliseconds
+ *
+ * qdf_mc_timer_get_system_time() function returns the number of milliseconds
+ * that have elapsed since the system was started
+ *
+ * Return:
+ * The current system time in milliseconds
+ */
+unsigned long qdf_mc_timer_get_system_time(void)
+{
+ struct timeval tv;
+ do_gettimeofday(&tv);
+ return tv.tv_sec * 1000 + tv.tv_usec / 1000;
+}
+EXPORT_SYMBOL(qdf_mc_timer_get_system_time);