os: Fix timer race conditions
Fixing following kind of race-conditions -
    WaitForSomething()
    |
    ---->  // timers -> timer-1 -> timer-2 -> null
           while (timers && (int) (timers->expires - now) <= 0)
               // prototype - DoTimer(OsTimerPtr timer, CARD32 now, OsTimerPtr *prev)
               DoTimer(timers, now, &timers)
               |
               |
               ----> OsBlockSignals();  .... OS Signal comes just before blocking it,
                                        .... timer-1 handler gets called.
                                             // timer-1 gets served and scheduled again;
                                             // timers -> timer-2 -> timer-1 -> null
                                        ....
                     *prev = timer->next;
                      timer->next = NULL;   // timers -> null
                      // timers list gets corrupted here and timer-2 gets removed from list.
Fixes: https://bugs.freedesktop.org/show_bug.cgi?id=86288
Signed-off-by: Nikhil Mahale <nmahale@nvidia.com>
Reviewed-by: Julien Cristau <jcristau@debian.org>
v2: Apply warning fixes from Keith Packard <keithp@keithp.com>
Reviewed-by: Aaron Plattner <aplattner@nvidia.com>
Signed-off-by: Aaron Plattner <aplattner@nvidia.com>
Signed-off-by: Keith Packard <keithp@keithp.com>
			
			
This commit is contained in:
		
							parent
							
								
									58f28b0427
								
							
						
					
					
						commit
						fe4c774c57
					
				
							
								
								
									
										41
									
								
								os/WaitFor.c
								
								
								
								
							
							
						
						
									
										41
									
								
								os/WaitFor.c
								
								
								
								
							|  | @ -121,9 +121,9 @@ struct _OsTimerRec { | ||||||
|     void *arg; |     void *arg; | ||||||
| }; | }; | ||||||
| 
 | 
 | ||||||
| static void DoTimer(OsTimerPtr timer, CARD32 now, OsTimerPtr *prev); | static void DoTimer(OsTimerPtr timer, CARD32 now, volatile OsTimerPtr *prev); | ||||||
| static void CheckAllTimers(void); | static void CheckAllTimers(void); | ||||||
| static OsTimerPtr timers = NULL; | static volatile OsTimerPtr timers = NULL; | ||||||
| 
 | 
 | ||||||
| /*****************
 | /*****************
 | ||||||
|  * WaitForSomething: |  * WaitForSomething: | ||||||
|  | @ -263,11 +263,14 @@ WaitForSomething(int *pClientsReady) | ||||||
|                 if ((int) (timers->expires - now) <= 0) |                 if ((int) (timers->expires - now) <= 0) | ||||||
|                     expired = 1; |                     expired = 1; | ||||||
| 
 | 
 | ||||||
|                 while (timers && (int) (timers->expires - now) <= 0) |                 if (expired) { | ||||||
|                     DoTimer(timers, now, &timers); |                     OsBlockSignals(); | ||||||
|  |                     while (timers && (int) (timers->expires - now) <= 0) | ||||||
|  |                         DoTimer(timers, now, &timers); | ||||||
|  |                     OsReleaseSignals(); | ||||||
| 
 | 
 | ||||||
|                 if (expired) |  | ||||||
|                     return 0; |                     return 0; | ||||||
|  |                 } | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
|         else { |         else { | ||||||
|  | @ -281,11 +284,14 @@ WaitForSomething(int *pClientsReady) | ||||||
|                     if ((int) (timers->expires - now) <= 0) |                     if ((int) (timers->expires - now) <= 0) | ||||||
|                         expired = 1; |                         expired = 1; | ||||||
| 
 | 
 | ||||||
|                     while (timers && (int) (timers->expires - now) <= 0) |                     if (expired) { | ||||||
|                         DoTimer(timers, now, &timers); |                         OsBlockSignals(); | ||||||
|  |                         while (timers && (int) (timers->expires - now) <= 0) | ||||||
|  |                             DoTimer(timers, now, &timers); | ||||||
|  |                         OsReleaseSignals(); | ||||||
| 
 | 
 | ||||||
|                     if (expired) |  | ||||||
|                         return 0; |                         return 0; | ||||||
|  |                     } | ||||||
|                 } |                 } | ||||||
|             } |             } | ||||||
|             if (someReady) |             if (someReady) | ||||||
|  | @ -401,24 +407,25 @@ CheckAllTimers(void) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| static void | static void | ||||||
| DoTimer(OsTimerPtr timer, CARD32 now, OsTimerPtr *prev) | DoTimer(OsTimerPtr timer, CARD32 now, volatile OsTimerPtr *prev) | ||||||
| { | { | ||||||
|     CARD32 newTime; |     CARD32 newTime; | ||||||
| 
 | 
 | ||||||
|     OsBlockSignals(); |     OsBlockSignals(); | ||||||
|     *prev = timer->next; |     *prev = timer->next; | ||||||
|     timer->next = NULL; |     timer->next = NULL; | ||||||
|  |     OsReleaseSignals(); | ||||||
|  | 
 | ||||||
|     newTime = (*timer->callback) (timer, now, timer->arg); |     newTime = (*timer->callback) (timer, now, timer->arg); | ||||||
|     if (newTime) |     if (newTime) | ||||||
|         TimerSet(timer, 0, newTime, timer->callback, timer->arg); |         TimerSet(timer, 0, newTime, timer->callback, timer->arg); | ||||||
|     OsReleaseSignals(); |  | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| OsTimerPtr | OsTimerPtr | ||||||
| TimerSet(OsTimerPtr timer, int flags, CARD32 millis, | TimerSet(OsTimerPtr timer, int flags, CARD32 millis, | ||||||
|          OsTimerCallback func, void *arg) |          OsTimerCallback func, void *arg) | ||||||
| { | { | ||||||
|     register OsTimerPtr *prev; |     volatile OsTimerPtr *prev; | ||||||
|     CARD32 now = GetTimeInMillis(); |     CARD32 now = GetTimeInMillis(); | ||||||
| 
 | 
 | ||||||
|     if (!timer) { |     if (!timer) { | ||||||
|  | @ -470,7 +477,7 @@ Bool | ||||||
| TimerForce(OsTimerPtr timer) | TimerForce(OsTimerPtr timer) | ||||||
| { | { | ||||||
|     int rc = FALSE; |     int rc = FALSE; | ||||||
|     OsTimerPtr *prev; |     volatile OsTimerPtr *prev; | ||||||
| 
 | 
 | ||||||
|     OsBlockSignals(); |     OsBlockSignals(); | ||||||
|     for (prev = &timers; *prev; prev = &(*prev)->next) { |     for (prev = &timers; *prev; prev = &(*prev)->next) { | ||||||
|  | @ -487,7 +494,7 @@ TimerForce(OsTimerPtr timer) | ||||||
| void | void | ||||||
| TimerCancel(OsTimerPtr timer) | TimerCancel(OsTimerPtr timer) | ||||||
| { | { | ||||||
|     OsTimerPtr *prev; |     volatile OsTimerPtr *prev; | ||||||
| 
 | 
 | ||||||
|     if (!timer) |     if (!timer) | ||||||
|         return; |         return; | ||||||
|  | @ -515,8 +522,12 @@ TimerCheck(void) | ||||||
| { | { | ||||||
|     CARD32 now = GetTimeInMillis(); |     CARD32 now = GetTimeInMillis(); | ||||||
| 
 | 
 | ||||||
|     while (timers && (int) (timers->expires - now) <= 0) |     if (timers && (int) (timers->expires - now) <= 0) { | ||||||
|         DoTimer(timers, now, &timers); |         OsBlockSignals(); | ||||||
|  |         while (timers && (int) (timers->expires - now) <= 0) | ||||||
|  |             DoTimer(timers, now, &timers); | ||||||
|  |         OsReleaseSignals(); | ||||||
|  |     } | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| void | void | ||||||
|  |  | ||||||
		Loading…
	
		Reference in New Issue