[svn-commits] mmichelson: trunk r168898 - /trunk/res/res_timing_timerfd.c

SVN commits to the Digium repositories svn-commits at lists.digium.com
Fri Jan 16 13:54:39 CST 2009


Author: mmichelson
Date: Fri Jan 16 13:54:39 2009
New Revision: 168898

URL: http://svn.digium.com/svn-view/asterisk?view=rev&rev=168898
Log:
Fix a logic error that occur when using the timerfd interface

This sequence of events posed a problem

timerfd_timer_open
timerfd_timer_enable_continuous
timerfd_timer_set_rate
timerfd_timer_disable_continuous

The reason was that the timing module was written under the assumption
that timerfd_timer_set_rate would not be called between enabling and
disabling continuous mode. What happened in this situation was that 
timerfd_timer_enable_continuous saved off our previously set timer (in this
situation a 0 timer, meaning it never runs out). Then timerfd_timer_disable_continuous
would restore this 0 timer, even though it logically should set the timer to be whatever
was set in timerfd_timer_set_rate.

Now the behavior in timerfd_timer_set_rate is to overwrite the saved timer that may
or may not have been set in timerfd_timer_enable_continuous. Even if
timerfd_timer_enable_continuous has not been previously called, this will not harm the
operation.

Thanks to Terry Wilson for discovering the problem and giving me a really great debug
capture that pointed out the problem clearly


Modified:
    trunk/res/res_timing_timerfd.c

Modified: trunk/res/res_timing_timerfd.c
URL: http://svn.digium.com/svn-view/asterisk/trunk/res/res_timing_timerfd.c?view=diff&rev=168898&r1=168897&r2=168898
==============================================================================
--- trunk/res/res_timing_timerfd.c (original)
+++ trunk/res/res_timing_timerfd.c Fri Jan 16 13:54:39 2009
@@ -131,13 +131,21 @@
 
 static int timerfd_timer_set_rate(int handle, unsigned int rate)
 {
-	struct itimerspec itspec;
-	itspec.it_value.tv_sec = 0;
-	itspec.it_value.tv_nsec = rate ? (long) (1000000000 / rate) : 0L;
-	itspec.it_interval.tv_sec = itspec.it_value.tv_sec;
-	itspec.it_interval.tv_nsec = itspec.it_value.tv_nsec;
-
-	return timerfd_settime(handle, 0, &itspec, NULL);
+	struct timerfd_timer *our_timer, find_helper = {
+		.handle = handle,
+	};
+
+	if (!(our_timer = ao2_find(timerfd_timers, &find_helper, OBJ_POINTER))) {
+		ast_log(LOG_ERROR, "Couldn't find timer with handle %d\n", handle);
+		return -1;
+	}
+
+	our_timer->saved_timer.it_value.tv_sec = 0;
+	our_timer->saved_timer.it_value.tv_nsec = rate ? (long) (1000000000 / rate) : 0L;
+	our_timer->saved_timer.it_interval.tv_sec = our_timer->saved_timer.it_value.tv_sec;
+	our_timer->saved_timer.it_interval.tv_nsec = our_timer->saved_timer.it_value.tv_nsec;
+
+	return timerfd_settime(handle, 0, &our_timer->saved_timer, NULL);
 }
 
 static void timerfd_timer_ack(int handle, unsigned int quantity)




More information about the svn-commits mailing list