2019-05-27 09:55:05 +03:00
// SPDX-License-Identifier: GPL-2.0-or-later
2010-09-21 04:42:46 +04:00
/*
2010-12-09 23:02:18 +03:00
* Generic Timer - queue
2010-09-21 04:42:46 +04:00
*
2010-12-09 23:02:18 +03:00
* Manages a simple queue of timers , ordered by expiration time .
2010-09-21 04:42:46 +04:00
* Uses rbtrees for quick list adds and expiration .
*
* NOTE : All of the following functions need to be serialized
2011-03-31 05:57:33 +04:00
* to avoid races . No locking is done by this library code .
2010-09-21 04:42:46 +04:00
*/
2012-01-21 03:35:53 +04:00
# include <linux/bug.h>
2010-12-09 23:02:18 +03:00
# include <linux/timerqueue.h>
2010-09-21 04:42:46 +04:00
# include <linux/rbtree.h>
2011-11-17 06:29:17 +04:00
# include <linux/export.h>
2010-09-21 04:42:46 +04:00
2020-04-29 18:07:53 +03:00
# define __node_2_tq(_n) \
rb_entry ( ( _n ) , struct timerqueue_node , node )
static inline bool __timerqueue_less ( struct rb_node * a , const struct rb_node * b )
{
return __node_2_tq ( a ) - > expires < __node_2_tq ( b ) - > expires ;
}
2010-09-21 04:42:46 +04:00
/**
2010-12-09 23:02:18 +03:00
* timerqueue_add - Adds timer to timerqueue .
2010-09-21 04:42:46 +04:00
*
2010-12-09 23:02:18 +03:00
* @ head : head of timerqueue
2010-09-21 04:42:46 +04:00
* @ node : timer node to be added
*
2017-12-22 17:51:15 +03:00
* Adds the timer node to the timerqueue , sorted by the node ' s expires
* value . Returns true if the newly added timer is the first expiring timer in
* the queue .
2010-09-21 04:42:46 +04:00
*/
2015-04-15 00:08:46 +03:00
bool timerqueue_add ( struct timerqueue_head * head , struct timerqueue_node * node )
2010-09-21 04:42:46 +04:00
{
/* Make sure we don't add nodes that are already added */
WARN_ON_ONCE ( ! RB_EMPTY_NODE ( & node - > node ) ) ;
2020-04-29 18:07:53 +03:00
return rb_add_cached ( & node - > node , & head - > rb_root , __timerqueue_less ) ;
2010-09-21 04:42:46 +04:00
}
2010-12-07 00:32:12 +03:00
EXPORT_SYMBOL_GPL ( timerqueue_add ) ;
2010-09-21 04:42:46 +04:00
/**
2010-12-09 23:02:18 +03:00
* timerqueue_del - Removes a timer from the timerqueue .
2010-09-21 04:42:46 +04:00
*
2010-12-09 23:02:18 +03:00
* @ head : head of timerqueue
2010-09-21 04:42:46 +04:00
* @ node : timer node to be removed
*
2017-12-22 17:51:15 +03:00
* Removes the timer node from the timerqueue . Returns true if the queue is
* not empty after the remove .
2010-09-21 04:42:46 +04:00
*/
2015-04-15 00:08:46 +03:00
bool timerqueue_del ( struct timerqueue_head * head , struct timerqueue_node * node )
2010-09-21 04:42:46 +04:00
{
WARN_ON_ONCE ( RB_EMPTY_NODE ( & node - > node ) ) ;
2019-07-24 18:23:23 +03:00
rb_erase_cached ( & node - > node , & head - > rb_root ) ;
2010-09-21 04:42:46 +04:00
RB_CLEAR_NODE ( & node - > node ) ;
2019-07-24 18:23:23 +03:00
return ! RB_EMPTY_ROOT ( & head - > rb_root . rb_root ) ;
2010-09-21 04:42:46 +04:00
}
2010-12-07 00:32:12 +03:00
EXPORT_SYMBOL_GPL ( timerqueue_del ) ;
2010-09-21 04:42:46 +04:00
/**
2010-12-09 23:02:18 +03:00
* timerqueue_iterate_next - Returns the timer after the provided timer
2010-09-21 04:42:46 +04:00
*
* @ node : Pointer to a timer .
*
* Provides the timer that is after the given node . This is used , when
* necessary , to iterate through the list of timers in a timer list
* without modifying the list .
*/
2010-12-09 23:02:18 +03:00
struct timerqueue_node * timerqueue_iterate_next ( struct timerqueue_node * node )
2010-09-21 04:42:46 +04:00
{
struct rb_node * next ;
if ( ! node )
return NULL ;
next = rb_next ( & node - > node ) ;
if ( ! next )
return NULL ;
2010-12-09 23:02:18 +03:00
return container_of ( next , struct timerqueue_node , node ) ;
2010-09-21 04:42:46 +04:00
}
2010-12-07 00:32:12 +03:00
EXPORT_SYMBOL_GPL ( timerqueue_iterate_next ) ;