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
/**
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
{
struct rb_node * * p = & head - > head . rb_node ;
struct rb_node * parent = NULL ;
2010-12-09 23:02:18 +03:00
struct timerqueue_node * ptr ;
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 ) ) ;
while ( * p ) {
parent = * p ;
2010-12-09 23:02:18 +03:00
ptr = rb_entry ( parent , struct timerqueue_node , node ) ;
2016-12-25 13:38:40 +03:00
if ( node - > expires < ptr - > expires )
2010-09-21 04:42:46 +04:00
p = & ( * p ) - > rb_left ;
else
p = & ( * p ) - > rb_right ;
}
rb_link_node ( & node - > node , parent , p ) ;
rb_insert_color ( & node - > node , & head - > head ) ;
2016-12-25 13:38:40 +03:00
if ( ! head - > next | | node - > expires < head - > next - > expires ) {
2010-09-21 04:42:46 +04:00
head - > next = node ;
2015-04-15 00:08:46 +03:00
return true ;
}
return false ;
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 ) ) ;
/* update next pointer */
if ( head - > next = = node ) {
struct rb_node * rbn = rb_next ( & node - > node ) ;
2016-12-20 16:57:44 +03:00
head - > next = rb_entry_safe ( rbn , struct timerqueue_node , node ) ;
2010-09-21 04:42:46 +04:00
}
rb_erase ( & node - > node , & head - > head ) ;
RB_CLEAR_NODE ( & node - > node ) ;
2015-04-15 00:08:46 +03:00
return head - > next ! = NULL ;
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 ) ;