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
*
* This program is free software ; you can redistribute it and / or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation ; either version 2 of the License , or
* ( at your option ) any later version .
*
* This program is distributed in the hope that it will be useful ,
* but WITHOUT ANY WARRANTY ; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE . See the
* GNU General Public License for more details .
*
* You should have received a copy of the GNU General Public License
* along with this program ; if not , write to the Free Software
* Foundation , Inc . , 59 Temple Place , Suite 330 , Boston , MA 02111 - 1307 USA
*/
2010-12-09 23:02:18 +03:00
# include <linux/timerqueue.h>
2010-09-21 04:42:46 +04:00
# include <linux/rbtree.h>
2010-12-07 00:32:12 +03:00
# include <linux/module.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
*
2010-12-09 23:02:18 +03:00
* Adds the timer node to the timerqueue , sorted by the
2010-09-21 04:42:46 +04:00
* node ' s expires value .
*/
2010-12-09 23:02:18 +03:00
void 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 ) ;
2010-09-21 04:42:46 +04:00
if ( node - > expires . tv64 < ptr - > expires . tv64 )
p = & ( * p ) - > rb_left ;
else
p = & ( * p ) - > rb_right ;
}
rb_link_node ( & node - > node , parent , p ) ;
rb_insert_color ( & node - > node , & head - > head ) ;
if ( ! head - > next | | node - > expires . tv64 < head - > next - > expires . tv64 )
head - > next = node ;
}
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
*
2010-12-09 23:02:18 +03:00
* Removes the timer node from the timerqueue .
2010-09-21 04:42:46 +04:00
*/
2010-12-09 23:02:18 +03:00
void 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 ) ;
head - > next = rbn ?
2010-12-09 23:02:18 +03:00
rb_entry ( rbn , struct timerqueue_node , node ) : NULL ;
2010-09-21 04:42:46 +04:00
}
rb_erase ( & node - > node , & head - > head ) ;
RB_CLEAR_NODE ( & node - > node ) ;
}
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 ) ;