• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /******************************************************************************
2  *
3  *  Copyright (C) 1999-2012 Broadcom Corporation
4  *
5  *  Licensed under the Apache License, Version 2.0 (the "License");
6  *  you may not use this file except in compliance with the License.
7  *  You may obtain a copy of the License at:
8  *
9  *  http://www.apache.org/licenses/LICENSE-2.0
10  *
11  *  Unless required by applicable law or agreed to in writing, software
12  *  distributed under the License is distributed on an "AS IS" BASIS,
13  *  WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
14  *  See the License for the specific language governing permissions and
15  *  limitations under the License.
16  *
17  ******************************************************************************/
18 #include <android-base/stringprintf.h>
19 #include <base/logging.h>
20 #include "gki_int.h"
21 
22 /* Make sure that this has been defined in target.h */
23 #ifndef GKI_NUM_TIMERS
24 #error NO TIMERS: Must define at least 1 timer in the system!
25 #endif
26 
27 /* Largest signed positive timer count */
28 #define GKI_NO_NEW_TMRS_STARTED (0x7fffffffL)
29 /* Marks an unused timer list entry (initial value) */
30 #define GKI_UNUSED_LIST_ENTRY (0x80000000L)
31 #define GKI_MAX_INT32 (0x7fffffffL)
32 
33 using android::base::StringPrintf;
34 
35 extern bool nfc_debug_enabled;
36 
37 /*******************************************************************************
38 **
39 ** Function         gki_timers_init
40 **
41 ** Description      This internal function is called once at startup to
42 **                  initialize all the timer structures.
43 **
44 ** Returns          void
45 **
46 *******************************************************************************/
gki_timers_init(void)47 void gki_timers_init(void) {
48   uint8_t tt;
49 
50   gki_cb.com.OSTicksTilExp =
51       0; /* Remaining time (of OSTimeCurTimeout) before next timer expires */
52   gki_cb.com.OSNumOrigTicks = 0;
53 #if (GKI_DELAY_STOP_SYS_TICK > 0)
54   gki_cb.com.OSTicksTilStop = 0; /* clear inactivity delay timer */
55 #endif
56 
57   for (tt = 0; tt < GKI_MAX_TASKS; tt++) {
58     gki_cb.com.OSWaitTmr[tt] = 0;
59 
60 #if (GKI_NUM_TIMERS > 0)
61     gki_cb.com.OSTaskTmr0[tt] = 0;
62     gki_cb.com.OSTaskTmr0R[tt] = 0;
63 #endif
64 
65 #if (GKI_NUM_TIMERS > 1)
66     gki_cb.com.OSTaskTmr1[tt] = 0;
67     gki_cb.com.OSTaskTmr1R[tt] = 0;
68 #endif
69 
70 #if (GKI_NUM_TIMERS > 2)
71     gki_cb.com.OSTaskTmr2[tt] = 0;
72     gki_cb.com.OSTaskTmr2R[tt] = 0;
73 #endif
74 
75 #if (GKI_NUM_TIMERS > 3)
76     gki_cb.com.OSTaskTmr3[tt] = 0;
77     gki_cb.com.OSTaskTmr3R[tt] = 0;
78 #endif
79   }
80 
81   for (tt = 0; tt < GKI_MAX_TIMER_QUEUES; tt++) {
82     gki_cb.com.timer_queues[tt] = NULL;
83   }
84 
85   gki_cb.com.p_tick_cb = NULL;
86   gki_cb.com.system_tick_running = false;
87 
88   return;
89 }
90 
91 /*******************************************************************************
92 **
93 ** Function         gki_timers_is_timer_running
94 **
95 ** Description      This internal function is called to test if any gki timer
96 **                  are running
97 **
98 **
99 ** Returns          TRUE if at least one time is running in the system, FALSE
100 **                  else.
101 **
102 *******************************************************************************/
gki_timers_is_timer_running(void)103 bool gki_timers_is_timer_running(void) {
104   uint8_t tt;
105   for (tt = 0; tt < GKI_MAX_TASKS; tt++) {
106 #if (GKI_NUM_TIMERS > 0)
107     if (gki_cb.com.OSTaskTmr0[tt]) {
108       return true;
109     }
110 #endif
111 
112 #if (GKI_NUM_TIMERS > 1)
113     if (gki_cb.com.OSTaskTmr1[tt]) {
114       return true;
115     }
116 #endif
117 
118 #if (GKI_NUM_TIMERS > 2)
119     if (gki_cb.com.OSTaskTmr2[tt]) {
120       return true;
121     }
122 #endif
123 
124 #if (GKI_NUM_TIMERS > 3)
125     if (gki_cb.com.OSTaskTmr3[tt]) {
126       return true;
127     }
128 #endif
129   }
130 
131   return false;
132 }
133 
134 /*******************************************************************************
135 **
136 ** Function         GKI_get_tick_count
137 **
138 ** Description      This function returns the current system ticks
139 **
140 ** Returns          The current number of system ticks
141 **
142 *******************************************************************************/
GKI_get_tick_count(void)143 uint32_t GKI_get_tick_count(void) { return gki_cb.com.OSTicks; }
144 
145 /*******************************************************************************
146 **
147 ** Function         GKI_ready_to_sleep
148 **
149 ** Description      This function returns the number of system ticks until the
150 **                  next timer will expire.  It is typically called by a power
151 **                  savings manager to find out how long it can have the system
152 **                  sleep before it needs to service the next entry.
153 **
154 ** Parameters:      None
155 **
156 ** Returns          Number of ticks til the next timer expires
157 **                  Note: The value is a signed  value.  This value should be
158 **                      compared to x > 0, to avoid misinterpreting negative
159 **                      tick values.
160 **
161 *******************************************************************************/
GKI_ready_to_sleep(void)162 int32_t GKI_ready_to_sleep(void) { return (gki_cb.com.OSTicksTilExp); }
163 
164 /*******************************************************************************
165 **
166 ** Function         GKI_start_timer
167 **
168 ** Description      An application can call this function to start one of
169 **                  it's four general purpose timers. Any of the four timers
170 **                  can be 1-shot or continuous. If a timer is already running,
171 **                  it will be reset to the new parameters.
172 **
173 ** Parameters       tnum            - (input) timer number to be started
174 **                                            (TIMER_0, TIMER_1, TIMER_2, or
175 **                                            TIMER_3)
176 **                  ticks           - (input) the number of system ticks til the
177 **                                            timer expires.
178 **                  is_continuous   - (input) TRUE if timer restarts
179 **                                            automatically, else FALSE if it is
180 **                                            a 'one-shot'.
181 **
182 ** Returns          void
183 **
184 *******************************************************************************/
GKI_start_timer(uint8_t tnum,int32_t ticks,bool is_continuous)185 void GKI_start_timer(uint8_t tnum, int32_t ticks, bool is_continuous) {
186   int32_t reload;
187   int32_t orig_ticks;
188   uint8_t task_id = GKI_get_taskid();
189   bool bad_timer = false;
190 
191   if (ticks <= 0) ticks = 1;
192 
193   orig_ticks = ticks; /* save the ticks in case adjustment is necessary */
194 
195   /* If continuous timer, set reload, else set it to 0 */
196   if (is_continuous)
197     reload = ticks;
198   else
199     reload = 0;
200 
201   GKI_disable();
202 
203   if (gki_timers_is_timer_running() == false) {
204 #if (GKI_DELAY_STOP_SYS_TICK > 0)
205     /* if inactivity delay timer is not running, start system tick */
206     if (gki_cb.com.OSTicksTilStop == 0) {
207 #endif
208       if (gki_cb.com.p_tick_cb) {
209         /* start system tick */
210         gki_cb.com.system_tick_running = true;
211         (gki_cb.com.p_tick_cb)(true);
212       }
213 #if (GKI_DELAY_STOP_SYS_TICK > 0)
214     } else {
215       /* clear inactivity delay timer */
216       gki_cb.com.OSTicksTilStop = 0;
217     }
218 #endif
219   }
220   /* Add the time since the last task timer update.
221   ** Note that this works when no timers are active since
222   ** both OSNumOrigTicks and OSTicksTilExp are 0.
223   */
224   if (GKI_MAX_INT32 - (gki_cb.com.OSNumOrigTicks - gki_cb.com.OSTicksTilExp) >
225       ticks) {
226     ticks += gki_cb.com.OSNumOrigTicks - gki_cb.com.OSTicksTilExp;
227   } else
228     ticks = GKI_MAX_INT32;
229 
230   switch (tnum) {
231 #if (GKI_NUM_TIMERS > 0)
232     case TIMER_0:
233       gki_cb.com.OSTaskTmr0R[task_id] = reload;
234       gki_cb.com.OSTaskTmr0[task_id] = ticks;
235       break;
236 #endif
237 
238 #if (GKI_NUM_TIMERS > 1)
239     case TIMER_1:
240       gki_cb.com.OSTaskTmr1R[task_id] = reload;
241       gki_cb.com.OSTaskTmr1[task_id] = ticks;
242       break;
243 #endif
244 
245 #if (GKI_NUM_TIMERS > 2)
246     case TIMER_2:
247       gki_cb.com.OSTaskTmr2R[task_id] = reload;
248       gki_cb.com.OSTaskTmr2[task_id] = ticks;
249       break;
250 #endif
251 
252 #if (GKI_NUM_TIMERS > 3)
253     case TIMER_3:
254       gki_cb.com.OSTaskTmr3R[task_id] = reload;
255       gki_cb.com.OSTaskTmr3[task_id] = ticks;
256       break;
257 #endif
258     default:
259       bad_timer = true; /* Timer number is bad, so do not use */
260   }
261 
262   /* Update the expiration timeout if a legitimate timer */
263   if (!bad_timer) {
264     /* Only update the timeout value if it is less than any other newly started
265      * timers */
266     gki_adjust_timer_count(orig_ticks);
267   }
268 
269   GKI_enable();
270 }
271 
272 /*******************************************************************************
273 **
274 ** Function         GKI_stop_timer
275 **
276 ** Description      An application can call this function to stop one of
277 **                  it's four general purpose timers. There is no harm in
278 **                  stopping a timer that is already stopped.
279 **
280 ** Parameters       tnum - (input) timer number to be started (TIMER_0,
281 **                                 TIMER_1, TIMER_2, or TIMER_3)
282 ** Returns          void
283 **
284 *******************************************************************************/
GKI_stop_timer(uint8_t tnum)285 void GKI_stop_timer(uint8_t tnum) {
286   uint8_t task_id = GKI_get_taskid();
287 
288   GKI_disable();
289 
290   switch (tnum) {
291 #if (GKI_NUM_TIMERS > 0)
292     case TIMER_0:
293       gki_cb.com.OSTaskTmr0R[task_id] = 0;
294       gki_cb.com.OSTaskTmr0[task_id] = 0;
295       break;
296 #endif
297 
298 #if (GKI_NUM_TIMERS > 1)
299     case TIMER_1:
300       gki_cb.com.OSTaskTmr1R[task_id] = 0;
301       gki_cb.com.OSTaskTmr1[task_id] = 0;
302       break;
303 #endif
304 
305 #if (GKI_NUM_TIMERS > 2)
306     case TIMER_2:
307       gki_cb.com.OSTaskTmr2R[task_id] = 0;
308       gki_cb.com.OSTaskTmr2[task_id] = 0;
309       break;
310 #endif
311 
312 #if (GKI_NUM_TIMERS > 3)
313     case TIMER_3:
314       gki_cb.com.OSTaskTmr3R[task_id] = 0;
315       gki_cb.com.OSTaskTmr3[task_id] = 0;
316       break;
317 #endif
318   }
319 
320   if (gki_timers_is_timer_running() == false) {
321     if (gki_cb.com.p_tick_cb) {
322 #if (GKI_DELAY_STOP_SYS_TICK > 0)
323       /* if inactivity delay timer is not running */
324       if ((gki_cb.com.system_tick_running) &&
325           (gki_cb.com.OSTicksTilStop == 0)) {
326         /* set inactivity delay timer */
327         /* when timer expires, system tick will be stopped */
328         gki_cb.com.OSTicksTilStop = GKI_DELAY_STOP_SYS_TICK;
329       }
330 #else
331       gki_cb.com.system_tick_running = false;
332       (gki_cb.com.p_tick_cb)(false); /* stop system tick */
333 #endif
334     }
335   }
336 
337   GKI_enable();
338 }
339 
340 /*******************************************************************************
341 **
342 ** Function         GKI_timer_update
343 **
344 ** Description      This function is called by an OS to drive the GKI's timers.
345 **                  It is typically called at every system tick to
346 **                  update the timers for all tasks, and check for timeouts.
347 **
348 **                  Note: It has been designed to also allow for variable tick
349 **                        updates so that systems with strict power savings
350 **                        requirements can have the update occur at variable
351 **                        intervals.
352 **
353 ** Parameters:      ticks_since_last_update - (input) This is the number of
354 **                  TICKS that have occurred since the last time
355 **                  GKI_timer_update was called.
356 **
357 ** Returns          void
358 **
359 *******************************************************************************/
GKI_timer_update(int32_t ticks_since_last_update)360 void GKI_timer_update(int32_t ticks_since_last_update) {
361   uint8_t task_id;
362   long next_expiration; /* Holds the next soonest expiration time after this
363                            update */
364 
365   /* Increment the number of ticks used for time stamps */
366   gki_cb.com.OSTicks += ticks_since_last_update;
367 
368   /* If any timers are running in any tasks, decrement the remaining time til
369    * the timer updates need to take place (next expiration occurs)
370    */
371   gki_cb.com.OSTicksTilExp -= ticks_since_last_update;
372 
373   /* Don't allow timer interrupt nesting */
374   if (gki_cb.com.timer_nesting) return;
375 
376   gki_cb.com.timer_nesting = 1;
377 
378 #if (GKI_DELAY_STOP_SYS_TICK > 0)
379   /* if inactivity delay timer is set and expired */
380   if (gki_cb.com.OSTicksTilStop) {
381     if (gki_cb.com.OSTicksTilStop <= (uint32_t)ticks_since_last_update) {
382       if (gki_cb.com.p_tick_cb) {
383         gki_cb.com.system_tick_running = false;
384         (gki_cb.com.p_tick_cb)(false); /* stop system tick */
385       }
386       gki_cb.com.OSTicksTilStop = 0; /* clear inactivity delay timer */
387       gki_cb.com.timer_nesting = 0;
388       return;
389     } else
390       gki_cb.com.OSTicksTilStop -= ticks_since_last_update;
391   }
392 #endif
393 
394   /* No need to update the ticks if no timeout has occurred */
395   if (gki_cb.com.OSTicksTilExp > 0) {
396     gki_cb.com.timer_nesting = 0;
397     return;
398   }
399 
400   GKI_disable();
401 
402   next_expiration = GKI_NO_NEW_TMRS_STARTED;
403 
404   /* If here then gki_cb.com.OSTicksTilExp <= 0. If negative, then increase
405      gki_cb.com.OSNumOrigTicks
406      to account for the difference so timer updates below are decremented by the
407      full number
408      of ticks. gki_cb.com.OSNumOrigTicks is reset at the bottom of this function
409      so changing this
410      value only affects the timer updates below
411    */
412   gki_cb.com.OSNumOrigTicks -= gki_cb.com.OSTicksTilExp;
413 
414   /* Check for OS Task Timers */
415   for (task_id = 0; task_id < GKI_MAX_TASKS; task_id++) {
416     if (gki_cb.com.OSRdyTbl[task_id] == TASK_DEAD) {
417       // task is shutdown do not try to service timers
418       continue;
419     }
420 
421     if (gki_cb.com.OSWaitTmr[task_id] > 0) /* If timer is running */
422     {
423       gki_cb.com.OSWaitTmr[task_id] -= gki_cb.com.OSNumOrigTicks;
424       if (gki_cb.com.OSWaitTmr[task_id] <= 0) {
425         /* Timer Expired */
426         gki_cb.com.OSRdyTbl[task_id] = TASK_READY;
427       }
428     }
429 
430 #if (GKI_NUM_TIMERS > 0)
431     /* If any timer is running, decrement */
432     if (gki_cb.com.OSTaskTmr0[task_id] > 0) {
433       gki_cb.com.OSTaskTmr0[task_id] -= gki_cb.com.OSNumOrigTicks;
434 
435       if (gki_cb.com.OSTaskTmr0[task_id] <= 0) {
436 /* Set Timer 0 Expired event mask and reload timer */
437 #if (GKI_TIMER_UPDATES_FROM_ISR == TRUE)
438         GKI_isend_event(task_id, TIMER_0_EVT_MASK);
439 #else
440         GKI_send_event(task_id, TIMER_0_EVT_MASK);
441 #endif
442         gki_cb.com.OSTaskTmr0[task_id] = gki_cb.com.OSTaskTmr0R[task_id];
443       }
444     }
445 
446     /* Check to see if this timer is the next one to expire */
447     if (gki_cb.com.OSTaskTmr0[task_id] > 0 &&
448         gki_cb.com.OSTaskTmr0[task_id] < next_expiration)
449       next_expiration = gki_cb.com.OSTaskTmr0[task_id];
450 #endif
451 
452 #if (GKI_NUM_TIMERS > 1)
453     /* If any timer is running, decrement */
454     if (gki_cb.com.OSTaskTmr1[task_id] > 0) {
455       gki_cb.com.OSTaskTmr1[task_id] -= gki_cb.com.OSNumOrigTicks;
456 
457       if (gki_cb.com.OSTaskTmr1[task_id] <= 0) {
458 /* Set Timer 1 Expired event mask and reload timer */
459 #if (GKI_TIMER_UPDATES_FROM_ISR == TRUE)
460         GKI_isend_event(task_id, TIMER_1_EVT_MASK);
461 #else
462         GKI_send_event(task_id, TIMER_1_EVT_MASK);
463 #endif
464         gki_cb.com.OSTaskTmr1[task_id] = gki_cb.com.OSTaskTmr1R[task_id];
465       }
466     }
467 
468     /* Check to see if this timer is the next one to expire */
469     if (gki_cb.com.OSTaskTmr1[task_id] > 0 &&
470         gki_cb.com.OSTaskTmr1[task_id] < next_expiration)
471       next_expiration = gki_cb.com.OSTaskTmr1[task_id];
472 #endif
473 
474 #if (GKI_NUM_TIMERS > 2)
475     /* If any timer is running, decrement */
476     if (gki_cb.com.OSTaskTmr2[task_id] > 0) {
477       gki_cb.com.OSTaskTmr2[task_id] -= gki_cb.com.OSNumOrigTicks;
478 
479       if (gki_cb.com.OSTaskTmr2[task_id] <= 0) {
480 /* Set Timer 2 Expired event mask and reload timer */
481 #if (GKI_TIMER_UPDATES_FROM_ISR == TRUE)
482         GKI_isend_event(task_id, TIMER_2_EVT_MASK);
483 #else
484         GKI_send_event(task_id, TIMER_2_EVT_MASK);
485 #endif
486         gki_cb.com.OSTaskTmr2[task_id] = gki_cb.com.OSTaskTmr2R[task_id];
487       }
488     }
489 
490     /* Check to see if this timer is the next one to expire */
491     if (gki_cb.com.OSTaskTmr2[task_id] > 0 &&
492         gki_cb.com.OSTaskTmr2[task_id] < next_expiration)
493       next_expiration = gki_cb.com.OSTaskTmr2[task_id];
494 #endif
495 
496 #if (GKI_NUM_TIMERS > 3)
497     /* If any timer is running, decrement */
498     if (gki_cb.com.OSTaskTmr3[task_id] > 0) {
499       gki_cb.com.OSTaskTmr3[task_id] -= gki_cb.com.OSNumOrigTicks;
500 
501       if (gki_cb.com.OSTaskTmr3[task_id] <= 0) {
502 /* Set Timer 3 Expired event mask and reload timer */
503 #if (GKI_TIMER_UPDATES_FROM_ISR == TRUE)
504         GKI_isend_event(task_id, TIMER_3_EVT_MASK);
505 #else
506         GKI_send_event(task_id, TIMER_3_EVT_MASK);
507 #endif
508         gki_cb.com.OSTaskTmr3[task_id] = gki_cb.com.OSTaskTmr3R[task_id];
509       }
510     }
511 
512     /* Check to see if this timer is the next one to expire */
513     if (gki_cb.com.OSTaskTmr3[task_id] > 0 &&
514         gki_cb.com.OSTaskTmr3[task_id] < next_expiration)
515       next_expiration = gki_cb.com.OSTaskTmr3[task_id];
516 #endif
517   }
518 
519   /* Set the next timer experation value if there is one to start */
520   if (next_expiration < GKI_NO_NEW_TMRS_STARTED) {
521     gki_cb.com.OSTicksTilExp = gki_cb.com.OSNumOrigTicks = next_expiration;
522   } else {
523     gki_cb.com.OSTicksTilExp = gki_cb.com.OSNumOrigTicks = 0;
524   }
525 
526   gki_cb.com.timer_nesting = 0;
527 
528   GKI_enable();
529 
530   return;
531 }
532 
533 /*******************************************************************************
534 **
535 ** Function         GKI_timer_queue_empty
536 **
537 ** Description      This function is called by applications to see whether the
538 **                  timer queue is empty
539 **
540 ** Parameters
541 **
542 ** Returns          bool
543 **
544 *******************************************************************************/
GKI_timer_queue_empty(void)545 bool GKI_timer_queue_empty(void) {
546   uint8_t tt;
547 
548   for (tt = 0; tt < GKI_MAX_TIMER_QUEUES; tt++) {
549     if (gki_cb.com.timer_queues[tt]) return false;
550   }
551 
552   return true;
553 }
554 
555 /*******************************************************************************
556 **
557 ** Function         GKI_timer_queue_register_callback
558 **
559 ** Description      This function is called by applications to register system
560 **                  tick start/stop callback for time queues
561 **
562 **
563 ** Parameters       p_callback - (input) pointer to the system tick callback
564 **
565 ** Returns          bool
566 **
567 *******************************************************************************/
GKI_timer_queue_register_callback(SYSTEM_TICK_CBACK * p_callback)568 void GKI_timer_queue_register_callback(SYSTEM_TICK_CBACK* p_callback) {
569   gki_cb.com.p_tick_cb = p_callback;
570 
571   return;
572 }
573 
574 /*******************************************************************************
575 **
576 ** Function         GKI_init_timer_list
577 **
578 ** Description      This function is called by applications when they
579 **                  want to initialize a timer list.
580 **
581 ** Parameters       p_timer_listq - (input) pointer to the timer list queue
582 **                                          object
583 **
584 ** Returns          void
585 **
586 *******************************************************************************/
GKI_init_timer_list(TIMER_LIST_Q * p_timer_listq)587 void GKI_init_timer_list(TIMER_LIST_Q* p_timer_listq) {
588   p_timer_listq->p_first = NULL;
589   p_timer_listq->p_last = NULL;
590   p_timer_listq->last_ticks = 0;
591 
592   return;
593 }
594 
595 /*******************************************************************************
596 **
597 ** Function         GKI_init_timer_list_entry
598 **
599 ** Description      This function is called by the applications when they
600 **                  want to initialize a timer list entry. This must be
601 **                  done prior to first use of the entry.
602 **
603 ** Parameters       p_tle - (input) pointer to a timer list queue entry
604 **
605 ** Returns          void
606 **
607 *******************************************************************************/
GKI_init_timer_list_entry(TIMER_LIST_ENT * p_tle)608 void GKI_init_timer_list_entry(TIMER_LIST_ENT* p_tle) {
609   p_tle->p_next = NULL;
610   p_tle->p_prev = NULL;
611   p_tle->ticks = GKI_UNUSED_LIST_ENTRY;
612   p_tle->in_use = false;
613 }
614 
615 /*******************************************************************************
616 **
617 ** Function         GKI_update_timer_list
618 **
619 ** Description      This function is called by the applications when they
620 **                  want to update a timer list. This should be at every
621 **                  timer list unit tick, e.g. once per sec, once per minute
622 **                  etc.
623 **
624 ** Parameters       p_timer_listq - (input) pointer to the timer list queue
625 **                  object
626 **                  num_units_since_last_update - (input) number of units since
627 **                  the last update (allows for variable unit update)
628 **
629 ** NOTE: The following timer list update routines should not be used for exact
630 **       time critical purposes.  The timer tasks should be used when exact
631 **       timing is needed.
632 **
633 ** Returns          the number of timers that have expired
634 **
635 *******************************************************************************/
GKI_update_timer_list(TIMER_LIST_Q * p_timer_listq,int32_t num_units_since_last_update)636 uint16_t GKI_update_timer_list(TIMER_LIST_Q* p_timer_listq,
637                                int32_t num_units_since_last_update) {
638   TIMER_LIST_ENT* p_tle;
639   uint16_t num_time_out = 0;
640   int32_t rem_ticks;
641   int32_t temp_ticks;
642 
643   p_tle = p_timer_listq->p_first;
644 
645   /* First, get the guys who have previously timed out */
646   /* Note that the tick value of the timers should always be '0' */
647   while ((p_tle) && (p_tle->ticks <= 0)) {
648     num_time_out++;
649     p_tle = p_tle->p_next;
650   }
651 
652   /* Timer entriy tick values are relative to the preceeding entry */
653   rem_ticks = num_units_since_last_update;
654 
655   /* Now, adjust remaining timer entries */
656   while ((p_tle != NULL) && (rem_ticks > 0)) {
657     temp_ticks = p_tle->ticks;
658     p_tle->ticks -= rem_ticks;
659 
660     /* See if this timer has just timed out */
661     if (p_tle->ticks <= 0) {
662       /* We set the number of ticks to '0' so that the legacy code
663        * that assumes a '0' or nonzero value will still work as coded. */
664       p_tle->ticks = 0;
665 
666       num_time_out++;
667     }
668 
669     rem_ticks -= temp_ticks; /* Decrement the remaining ticks to process */
670     p_tle = p_tle->p_next;
671   }
672 
673   if (p_timer_listq->last_ticks > 0) {
674     p_timer_listq->last_ticks -= num_units_since_last_update;
675 
676     /* If the last timer has expired set last_ticks to 0 so that other list
677     * update
678     * functions will calculate correctly
679     */
680     if (p_timer_listq->last_ticks < 0) p_timer_listq->last_ticks = 0;
681   }
682 
683   return (num_time_out);
684 }
685 
686 /*******************************************************************************
687 **
688 ** Function         GKI_get_remaining_ticks
689 **
690 ** Description      This function is called by an application to get remaining
691 **                  ticks to expire
692 **
693 ** Parameters       p_timer_listq - (input) pointer to the timer list queue
694 **                                          object
695 **                  p_target_tle - (input) pointer to a timer list queue entry
696 **
697 ** Returns          0 if timer is not used or timer is not in the list
698 **                  remaining ticks if success
699 **
700 *******************************************************************************/
GKI_get_remaining_ticks(TIMER_LIST_Q * p_timer_listq,TIMER_LIST_ENT * p_target_tle)701 uint32_t GKI_get_remaining_ticks(TIMER_LIST_Q* p_timer_listq,
702                                  TIMER_LIST_ENT* p_target_tle) {
703   TIMER_LIST_ENT* p_tle;
704   uint32_t rem_ticks = 0;
705 
706   if (p_target_tle->in_use) {
707     p_tle = p_timer_listq->p_first;
708 
709     /* adding up all of ticks in previous entries */
710     while ((p_tle) && (p_tle != p_target_tle)) {
711       rem_ticks += p_tle->ticks;
712       p_tle = p_tle->p_next;
713     }
714 
715     /* if found target entry */
716     if (p_tle == p_target_tle) {
717       rem_ticks += p_tle->ticks;
718     } else {
719       LOG(ERROR) << StringPrintf(
720           "GKI_get_remaining_ticks: No timer entry in the list");
721       return (0);
722     }
723   } else {
724     LOG(ERROR) << StringPrintf(
725         "GKI_get_remaining_ticks: timer entry is not active");
726   }
727 
728   return (rem_ticks);
729 }
730 
731 /*******************************************************************************
732 **
733 ** Function         GKI_add_to_timer_list
734 **
735 ** Description      This function is called by an application to add a timer
736 **                  entry to a timer list.
737 **
738 **                  Note: A timer value of '0' will effectively insert an
739 **                        already expired event.  Negative tick values will be
740 **                        ignored.
741 **
742 ** Parameters       p_timer_listq - (input) pointer to the timer list queue
743 **                                          object
744 **                  p_tle - (input) pointer to a timer list queue entry
745 **
746 ** Returns          void
747 **
748 *******************************************************************************/
GKI_add_to_timer_list(TIMER_LIST_Q * p_timer_listq,TIMER_LIST_ENT * p_tle)749 void GKI_add_to_timer_list(TIMER_LIST_Q* p_timer_listq, TIMER_LIST_ENT* p_tle) {
750   uint32_t nr_ticks_total;
751   uint8_t tt;
752   TIMER_LIST_ENT* p_temp;
753   if (p_tle == NULL || p_timer_listq == NULL) {
754     DLOG_IF(INFO, nfc_debug_enabled) << StringPrintf(
755         "%s: invalid argument %p, %p****************************<<", __func__,
756         p_timer_listq, p_tle);
757     return;
758   }
759 
760   /* Only process valid tick values */
761   if (p_tle->ticks >= 0) {
762     /* If this entry is the last in the list */
763     if (p_tle->ticks >= p_timer_listq->last_ticks) {
764       /* If this entry is the only entry in the list */
765       if (p_timer_listq->p_first == NULL)
766         p_timer_listq->p_first = p_tle;
767       else {
768         /* Insert the entry onto the end of the list */
769         if (p_timer_listq->p_last != NULL)
770           p_timer_listq->p_last->p_next = p_tle;
771 
772         p_tle->p_prev = p_timer_listq->p_last;
773       }
774 
775       p_tle->p_next = NULL;
776       p_timer_listq->p_last = p_tle;
777       nr_ticks_total = p_tle->ticks;
778       p_tle->ticks -= p_timer_listq->last_ticks;
779 
780       p_timer_listq->last_ticks = nr_ticks_total;
781     } else /* This entry needs to be inserted before the last entry */
782     {
783       /* Find the entry that the new one needs to be inserted in front of */
784       p_temp = p_timer_listq->p_first;
785       while (p_tle->ticks > p_temp->ticks) {
786         /* Update the tick value if looking at an unexpired entry */
787         if (p_temp->ticks > 0) p_tle->ticks -= p_temp->ticks;
788 
789         p_temp = p_temp->p_next;
790       }
791 
792       /* The new entry is the first in the list */
793       if (p_temp == p_timer_listq->p_first) {
794         p_tle->p_next = p_timer_listq->p_first;
795         p_timer_listq->p_first->p_prev = p_tle;
796         p_timer_listq->p_first = p_tle;
797       } else {
798         p_temp->p_prev->p_next = p_tle;
799         p_tle->p_prev = p_temp->p_prev;
800         p_temp->p_prev = p_tle;
801         p_tle->p_next = p_temp;
802       }
803       p_temp->ticks -= p_tle->ticks;
804     }
805 
806     p_tle->in_use = true;
807 
808     /* if we already add this timer queue to the array */
809     for (tt = 0; tt < GKI_MAX_TIMER_QUEUES; tt++) {
810       if (gki_cb.com.timer_queues[tt] == p_timer_listq) return;
811     }
812     /* add this timer queue to the array */
813     for (tt = 0; tt < GKI_MAX_TIMER_QUEUES; tt++) {
814       if (gki_cb.com.timer_queues[tt] == NULL) break;
815     }
816     if (tt < GKI_MAX_TIMER_QUEUES) {
817       gki_cb.com.timer_queues[tt] = p_timer_listq;
818     }
819   }
820 
821   return;
822 }
823 
824 /*******************************************************************************
825 **
826 ** Function         GKI_remove_from_timer_list
827 **
828 ** Description      This function is called by an application to remove a timer
829 **                  entry from a timer list.
830 **
831 ** Parameters       p_timer_listq  - (input) pointer to the timer list queue
832 **                                            object
833 **                  p_tle - (input) pointer to a timer list queue entry
834 **
835 ** Returns          void
836 **
837 *******************************************************************************/
GKI_remove_from_timer_list(TIMER_LIST_Q * p_timer_listq,TIMER_LIST_ENT * p_tle)838 void GKI_remove_from_timer_list(TIMER_LIST_Q* p_timer_listq,
839                                 TIMER_LIST_ENT* p_tle) {
840   uint8_t tt;
841 
842   /* Verify that the entry is valid */
843   if (p_tle == NULL || p_tle->in_use == false ||
844       p_timer_listq->p_first == NULL) {
845     return;
846   }
847 
848   /* Add the ticks remaining in this timer (if any) to the next guy in the list.
849   ** Note: Expired timers have a tick value of '0'.
850   */
851   if (p_tle->p_next != NULL) {
852     p_tle->p_next->ticks += p_tle->ticks;
853   } else {
854     p_timer_listq->last_ticks -= p_tle->ticks;
855   }
856 
857   /* Unlink timer from the list.
858   */
859   if (p_timer_listq->p_first == p_tle) {
860     p_timer_listq->p_first = p_tle->p_next;
861 
862     if (p_timer_listq->p_first != NULL) p_timer_listq->p_first->p_prev = NULL;
863 
864     if (p_timer_listq->p_last == p_tle) p_timer_listq->p_last = NULL;
865   } else {
866     if (p_timer_listq->p_last == p_tle) {
867       p_timer_listq->p_last = p_tle->p_prev;
868 
869       if (p_timer_listq->p_last != NULL) p_timer_listq->p_last->p_next = NULL;
870     } else {
871       if (p_tle->p_next != NULL && p_tle->p_next->p_prev == p_tle)
872         p_tle->p_next->p_prev = p_tle->p_prev;
873       else {
874         /* Error case - chain messed up ?? */
875         return;
876       }
877 
878       if (p_tle->p_prev != NULL && p_tle->p_prev->p_next == p_tle)
879         p_tle->p_prev->p_next = p_tle->p_next;
880       else {
881         /* Error case - chain messed up ?? */
882         return;
883       }
884     }
885   }
886 
887   p_tle->p_next = p_tle->p_prev = NULL;
888   p_tle->ticks = GKI_UNUSED_LIST_ENTRY;
889   p_tle->in_use = false;
890 
891   /* if timer queue is empty */
892   if (p_timer_listq->p_first == NULL && p_timer_listq->p_last == NULL) {
893     for (tt = 0; tt < GKI_MAX_TIMER_QUEUES; tt++) {
894       if (gki_cb.com.timer_queues[tt] == p_timer_listq) {
895         gki_cb.com.timer_queues[tt] = NULL;
896         break;
897       }
898     }
899   }
900 
901   return;
902 }
903 
904 /*******************************************************************************
905 **
906 ** Function         gki_adjust_timer_count
907 **
908 ** Description      This function is called whenever a new timer or GKI_wait
909 **                  occurs to adjust (if necessary) the current time til the
910 **                  first expiration. This only needs to make an adjustment if
911 **                  the new timer (in ticks) is less than the number of ticks
912 **                  remaining on the current timer.
913 **
914 ** Parameters:      ticks - (input) number of system ticks of the new timer
915 **                                  entry
916 **
917 **                  NOTE:  This routine MUST be called while interrupts are
918 **                         disabled to avoid updates while adjusting the timer
919 **                         variables.
920 **
921 ** Returns          void
922 **
923 *******************************************************************************/
gki_adjust_timer_count(int32_t ticks)924 void gki_adjust_timer_count(int32_t ticks) {
925   if (ticks > 0) {
926     /* See if the new timer expires before the current first expiration */
927     if (gki_cb.com.OSNumOrigTicks == 0 ||
928         (ticks < gki_cb.com.OSTicksTilExp && gki_cb.com.OSTicksTilExp > 0)) {
929       gki_cb.com.OSNumOrigTicks =
930           (gki_cb.com.OSNumOrigTicks - gki_cb.com.OSTicksTilExp) + ticks;
931       gki_cb.com.OSTicksTilExp = ticks;
932     }
933   }
934 
935   return;
936 }
937