2019-05-27 08:55:05 +02:00
// SPDX-License-Identifier: GPL-2.0-or-later
2010-09-20 17:42:46 -07:00
/*
2010-12-09 12:02:18 -08:00
* Generic Timer - queue
2010-09-20 17:42:46 -07:00
*
2010-12-09 12:02:18 -08:00
* Manages a simple queue of timers , ordered by expiration time .
2010-09-20 17:42:46 -07:00
* Uses rbtrees for quick list adds and expiration .
*
* NOTE : All of the following functions need to be serialized
2011-03-30 22:57:33 -03:00
* to avoid races . No locking is done by this library code .
2010-09-20 17:42:46 -07:00
*/
2012-01-20 18:35:53 -05:00
# include <linux/bug.h>
2010-12-09 12:02:18 -08:00
# include <linux/timerqueue.h>
2010-09-20 17:42:46 -07:00
# include <linux/rbtree.h>
2011-11-16 21:29:17 -05:00
# include <linux/export.h>
2010-09-20 17:42:46 -07:00
/**
2010-12-09 12:02:18 -08:00
* timerqueue_add - Adds timer to timerqueue .
2010-09-20 17:42:46 -07:00
*
2010-12-09 12:02:18 -08:00
* @ head : head of timerqueue
2010-09-20 17:42:46 -07:00
* @ node : timer node to be added
*
2017-12-22 15:51:15 +01: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-20 17:42:46 -07:00
*/
2015-04-14 21:08:46 +00:00
bool timerqueue_add ( struct timerqueue_head * head , struct timerqueue_node * node )
2010-09-20 17:42:46 -07:00
{
2019-07-24 08:23:23 -07:00
struct rb_node * * p = & head - > rb_root . rb_root . rb_node ;
2010-09-20 17:42:46 -07:00
struct rb_node * parent = NULL ;
2019-07-24 08:23:23 -07:00
struct timerqueue_node * ptr ;
bool leftmost = true ;
2010-09-20 17:42:46 -07: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 12:02:18 -08:00
ptr = rb_entry ( parent , struct timerqueue_node , node ) ;
2019-07-24 08:23:23 -07:00
if ( node - > expires < ptr - > expires ) {
2010-09-20 17:42:46 -07:00
p = & ( * p ) - > rb_left ;
2019-07-24 08:23:23 -07:00
} else {
2010-09-20 17:42:46 -07:00
p = & ( * p ) - > rb_right ;
2019-07-24 08:23:23 -07:00
leftmost = false ;
}
2010-09-20 17:42:46 -07:00
}
rb_link_node ( & node - > node , parent , p ) ;
2019-07-24 08:23:23 -07:00
rb_insert_color_cached ( & node - > node , & head - > rb_root , leftmost ) ;
2010-09-20 17:42:46 -07:00
2019-07-24 08:23:23 -07:00
return leftmost ;
2010-09-20 17:42:46 -07:00
}
2010-12-06 13:32:12 -08:00
EXPORT_SYMBOL_GPL ( timerqueue_add ) ;
2010-09-20 17:42:46 -07:00
/**
2010-12-09 12:02:18 -08:00
* timerqueue_del - Removes a timer from the timerqueue .
2010-09-20 17:42:46 -07:00
*
2010-12-09 12:02:18 -08:00
* @ head : head of timerqueue
2010-09-20 17:42:46 -07:00
* @ node : timer node to be removed
*
2017-12-22 15:51:15 +01:00
* Removes the timer node from the timerqueue . Returns true if the queue is
* not empty after the remove .
2010-09-20 17:42:46 -07:00
*/
2015-04-14 21:08:46 +00:00
bool timerqueue_del ( struct timerqueue_head * head , struct timerqueue_node * node )
2010-09-20 17:42:46 -07:00
{
WARN_ON_ONCE ( RB_EMPTY_NODE ( & node - > node ) ) ;
2019-07-24 08:23:23 -07:00
rb_erase_cached ( & node - > node , & head - > rb_root ) ;
2010-09-20 17:42:46 -07:00
RB_CLEAR_NODE ( & node - > node ) ;
2019-07-24 08:23:23 -07:00
return ! RB_EMPTY_ROOT ( & head - > rb_root . rb_root ) ;
2010-09-20 17:42:46 -07:00
}
2010-12-06 13:32:12 -08:00
EXPORT_SYMBOL_GPL ( timerqueue_del ) ;
2010-09-20 17:42:46 -07:00
/**
2010-12-09 12:02:18 -08:00
* timerqueue_iterate_next - Returns the timer after the provided timer
2010-09-20 17:42:46 -07: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 12:02:18 -08:00
struct timerqueue_node * timerqueue_iterate_next ( struct timerqueue_node * node )
2010-09-20 17:42:46 -07:00
{
struct rb_node * next ;
if ( ! node )
return NULL ;
next = rb_next ( & node - > node ) ;
if ( ! next )
return NULL ;
2010-12-09 12:02:18 -08:00
return container_of ( next , struct timerqueue_node , node ) ;
2010-09-20 17:42:46 -07:00
}
2010-12-06 13:32:12 -08:00
EXPORT_SYMBOL_GPL ( timerqueue_iterate_next ) ;