AndroidO bluedroid alarm 機制分析

bluedroid的alarm 機制實如今osi/osi/src/alarm.cc 中:node

這裏面實現了不少的接口:react

alarm_t* alarm_new(const char* name);
alarm_t* alarm_new_periodic(const char* name) ;
static alarm_t* alarm_new_internal(const char* name, bool is_periodic) ;
void alarm_free(alarm_t* alarm);
void alarm_set(alarm_t* alarm, period_ms_t interval_ms, alarm_callback_t cb,
               void* data) ;
void alarm_cancel(alarm_t* alarm);
void alarm_cleanup(void);
static bool lazy_initialize(void) ;

 咱們先看一個使用alarm 的事例:app

在hci_layer.cc 文件中關於hci_module_start_up 的實現:less

 

 startup_timer = alarm_new("hci.startup_timer");
...
 alarm_set(startup_timer, startup_timeout_ms, startup_timer_expired, NULL); 

 當startup_timeout_ms 時間到達的時候,若是startup_timer尚未被取消的話,那麼startup_timer_expired函數將會被執行。函數

咱們具體看看 其實現的過程:post

咱們先看看新建定時器的代碼:ui

alarm_t* alarm_new(const char* name) { return alarm_new_internal(name, false); }
static alarm_t* alarm_new_internal(const char* name, bool is_periodic) {
  // Make sure we have a list we can insert alarms into.
  if (!alarms && !lazy_initialize()) { //最初alarm 尚未初始化,須要執行初始化流程
    CHECK(false);  // if initialization failed, we should not continue
    return NULL;
  }

  alarm_t* ret = static_cast<alarm_t*>(osi_calloc(sizeof(alarm_t)));

  ret->callback_mutex = new std::recursive_mutex;
  ret->is_periodic = is_periodic;
  ret->stats.name = osi_strdup(name);
  // NOTE: The stats were reset by osi_calloc() above

  return ret;
}

 

咱們繼續看看 lazy_initialize  是如何 初始化的:this

static bool lazy_initialize(void) {

  bool timer_initialized = false;
  bool wakeup_timer_initialized = false;

  std::lock_guard<std::mutex> lock(alarms_mutex);

  alarms = list_new(NULL);

  if (!timer_create_internal(CLOCK_ID, &timer)) goto error;//timer_create
  timer_initialized = true;

  if (!timer_create_internal(CLOCK_ID_ALARM, &wakeup_timer)) goto error;
  wakeup_timer_initialized = true;

  alarm_expired = semaphore_new(0);//新建信號量

  default_callback_thread =
      thread_new_sized("alarm_default_callbacks", SIZE_MAX);//新建線程

  thread_set_rt_priority(default_callback_thread, THREAD_RT_PRIORITY);//提升線程優先級
  default_callback_queue = fixed_queue_new(SIZE_MAX);//新建隊列

  alarm_register_processing_queue(default_callback_queue,
                                  default_callback_thread);//將線程和隊列綁定

  dispatcher_thread_active = true;
  dispatcher_thread = thread_new("alarm_dispatcher");//新建定時器分發線程,該線程不停運行while(true)

  thread_set_rt_priority(dispatcher_thread, THREAD_RT_PRIORITY);
  thread_post(dispatcher_thread, callback_dispatch, NULL);//運行callback_dispatch
  return true;

error:
  ...
  return false;
}

 這裏有兩個線程,一個是dispatcher_thread,它一直輪詢,若是有定時器到期那麼他就將這個定時器放到一個(當初建立啊定時器的時候關聯的隊列)特定的隊列裏面,默認的就是default_callback_queue ,而後由另一個線程default_callback_thread 來處理該隊列裏面的已經expire的定時器.lua

咱們看看timer_create_internal的實現:spa

static bool timer_create_internal(const clockid_t clock_id, timer_t* timer) {
  CHECK(timer != NULL);

  struct sigevent sigevent;
  // create timer with RT priority thread
  pthread_attr_t thread_attr;
  pthread_attr_init(&thread_attr);
  pthread_attr_setschedpolicy(&thread_attr, SCHED_FIFO);
  struct sched_param param;
  param.sched_priority = THREAD_RT_PRIORITY;
  pthread_attr_setschedparam(&thread_attr, &param);

  memset(&sigevent, 0, sizeof(sigevent));
  sigevent.sigev_notify = SIGEV_THREAD;
  sigevent.sigev_notify_function = (void (*)(union sigval))timer_callback;//發送信號量
  sigevent.sigev_notify_attributes = &thread_attr;
  if (timer_create(clock_id, &sigevent, timer) == -1) {  //建立定時器
    /*錯誤處理*/
    }
    return false;
  }

  return true;
}

 

這裏咱們注意到,當定時器到期的時候,會執行timer_callback,其就是發送了一個alarm_expired的信號量:

static void timer_callback(UNUSED_ATTR void* ptr) {
  semaphore_post(alarm_expired);
}

 

這裏有發送信號量,那麼必定有一個地方會等待這個信號量,就是在定時器的不斷等待的線程裏面:

static void callback_dispatch(UNUSED_ATTR void* context) {
  while (true) {
    semaphore_wait(alarm_expired);//一直循環等待信號量
...
    }
}

 

 lazy_initialize   初始化完成以後,定時器並無啓動,只是建立了定時器.那是在哪裏啓動的呢?根據咱們上面展現出的使用示例,在alarm_set 確定是有啓動定時器的操做的:

void alarm_set(alarm_t* alarm, period_ms_t interval_ms, alarm_callback_t cb,
               void* data) {
  alarm_set_on_queue(alarm, interval_ms, cb, data, default_callback_queue);
}

 

這裏發現 調用alarm_set 傳入的 隊列都是default_callback_queue,

void alarm_set_on_queue(alarm_t* alarm, period_ms_t interval_ms,
                        alarm_callback_t cb, void* data, fixed_queue_t* queue) {
  CHECK(queue != NULL);
  alarm_set_internal(alarm, interval_ms, cb, data, queue);
}

 

// Runs in exclusion with alarm_cancel and timer_callback.
static void alarm_set_internal(alarm_t* alarm, period_ms_t period,
                               alarm_callback_t cb, void* data,
                               fixed_queue_t* queue) {

  std::lock_guard<std::mutex> lock(alarms_mutex);

  alarm->creation_time = now();
  alarm->period = period;
  alarm->queue = queue;
  alarm->callback = cb;
  alarm->data = data;

  schedule_next_instance(alarm);
  alarm->stats.scheduled_count++;
}

 

上面是對定時器進行封裝以及賦值,而後調用schedule_next_instance 來啓動 定時器:

 

// Must be called with |alarms_mutex| held
static void schedule_next_instance(alarm_t* alarm) {
  // If the alarm is currently set and it's at the start of the list,
  // we'll need to re-schedule since we've adjusted the earliest deadline.
  bool needs_reschedule =
      (!list_is_empty(alarms) && list_front(alarms) == alarm);//若是alarms 隊列第一個元素就是這個定時器,那麼須要重啓schedule
  if (alarm->callback) remove_pending_alarm(alarm);//取出全部的pending,重複的alarm

  // Calculate the next deadline for this alarm
  period_ms_t just_now = now();
  period_ms_t ms_into_period = 0;
  if ((alarm->is_periodic) && (alarm->period != 0))
    ms_into_period = ((just_now - alarm->creation_time) % alarm->period);
  alarm->deadline = just_now + (alarm->period - ms_into_period);

  // Add it into the timer list sorted by deadline (earliest deadline first).//如下是給alarm排序,插入到某個合適的問題,最近的alarm 排在第一個
  if (list_is_empty(alarms) ||
      ((alarm_t*)list_front(alarms))->deadline > alarm->deadline) {
    list_prepend(alarms, alarm);
  } else {
    for (list_node_t* node = list_begin(alarms); node != list_end(alarms);
         node = list_next(node)) {
      list_node_t* next = list_next(node);
      if (next == list_end(alarms) ||
          ((alarm_t*)list_node(next))->deadline > alarm->deadline) {
        list_insert_after(alarms, node, alarm);
        break;
      }
    }
  }

  // If the new alarm has the earliest deadline, we need to re-evaluate our
  // schedule.
  if (needs_reschedule ||
      (!list_is_empty(alarms) && list_front(alarms) == alarm)) {
    reschedule_root_alarm();
  }
}

 

上面主要就是將alarm插入到 alarms 列表中,等待schedule,若是當前這個alarm 就是最緊迫的alarm,那麼就會當即進行 schedule.

咱們看看其實現reschedule_root_alarm;

// NOTE: must be called with |alarms_mutex| held
static void reschedule_root_alarm(void) {
  CHECK(alarms != NULL);

  const bool timer_was_set = timer_set;
  alarm_t* next;
  int64_t next_expiration;

  // If used in a zeroed state, disarms the timer.
  struct itimerspec timer_time;
  memset(&timer_time, 0, sizeof(timer_time));

  if (list_is_empty(alarms)) goto done;

  next = static_cast<alarm_t*>(list_front(alarms));
  next_expiration = next->deadline - now();
  if (next_expiration < TIMER_INTERVAL_FOR_WAKELOCK_IN_MS) {//若是deadline<3s>

    timer_time.it_value.tv_sec = (next->deadline / 1000);
    timer_time.it_value.tv_nsec = (next->deadline % 1000) * 1000000LL;

/*下面設置最長的wake_up 是爲了減小刪除該timer的開銷,能夠略過*/
    struct itimerspec end_of_time;
    memset(&end_of_time, 0, sizeof(end_of_time));
    end_of_time.it_value.tv_sec = (time_t)(1LL << (sizeof(time_t) * 8 - 2));
    timer_settime(wakeup_timer, TIMER_ABSTIME, &end_of_time, NULL);
  } else {
    // WARNING: do not attempt to use relative timers with *_ALARM clock IDs
    // in kernels before 3.17 unless you have the following patch:
    // https://lkml.org/lkml/2014/7/7/576
    struct itimerspec wakeup_time;
    memset(&wakeup_time, 0, sizeof(wakeup_time));

    wakeup_time.it_value.tv_sec = (next->deadline / 1000);
    wakeup_time.it_value.tv_nsec = (next->deadline % 1000) * 1000000LL;
    if (timer_settime(wakeup_timer, TIMER_ABSTIME, &wakeup_time, NULL) == -1)
      LOG_ERROR(LOG_TAG, "%s unable to set wakeup timer: %s", __func__,
                strerror(errno));
  }

done:
  timer_set =
      timer_time.it_value.tv_sec != 0 || timer_time.it_value.tv_nsec != 0;
  if (timer_was_set && !timer_set) {
    wakelock_release();
  }

  if (timer_settime(timer, TIMER_ABSTIME, &timer_time, NULL) == -1)
    LOG_ERROR(LOG_TAG, "%s unable to set timer: %s", __func__, strerror(errno));//設置定時器

  // If next expiration was in the past (e.g. short timer that got context
  // switched) then the timer might have diarmed itself. Detect this case and
  // work around it by manually signalling the |alarm_expired| semaphore.
  //
  // It is possible that the timer was actually super short (a few
  // milliseconds) and the timer expired normally before we called
  // |timer_gettime|. Worst case, |alarm_expired| is signaled twice for that
  // alarm. Nothing bad should happen in that case though since the callback
  // dispatch function checks to make sure the timer at the head of the list
  // actually expired.
  if (timer_set) {
    struct itimerspec time_to_expire;
    timer_gettime(timer, &time_to_expire);
    if (time_to_expire.it_value.tv_sec == 0 &&
        time_to_expire.it_value.tv_nsec == 0) {

      semaphore_post(alarm_expired);//若是定時器的時機已經到了,那麼直接發送信號量
    }
  }
}

 

代碼是實現是 在離expire 不到3s的時候啓動定時器.

當定時器時間到的時候,發動alarm_expired的信號.

咱們接下來看看 定時器的 已經到期的處理流程:上面咱們已經知道,線程dispatcher_thread一直輪詢,咱們看看其實現:

// Function running on |dispatcher_thread| that performs the following:
//   (1) Receives a signal using |alarm_exired| that the alarm has expired
//   (2) Dispatches the alarm callback for processing by the corresponding
// thread for that alarm.
static void callback_dispatch(UNUSED_ATTR void* context) {
  while (true) {
    semaphore_wait(alarm_expired);//等待expire的信號量
    if (!dispatcher_thread_active) break;

    std::lock_guard<std::mutex> lock(alarms_mutex);
    alarm_t* alarm;

    // Take into account that the alarm may get cancelled before we get to it.
    // We're done here if there are no alarms or the alarm at the front is in
    // the future. Exit right away since there's nothing left to do.
    if (list_is_empty(alarms) ||
        (alarm = static_cast<alarm_t*>(list_front(alarms)))->deadline > now()) {
      reschedule_root_alarm();
      continue;
    }

    list_remove(alarms, alarm);//remove 該alarm 從隊列中

    if (alarm->is_periodic) {
      alarm->prev_deadline = alarm->deadline;
      schedule_next_instance(alarm);
      alarm->stats.rescheduled_count++;
    }
    reschedule_root_alarm();//去啓動下一個定時器

    // Enqueue the alarm for processing
    fixed_queue_enqueue(alarm->queue, alarm);//將該expire的定時器放到相應隊列等待處理
  }

  LOG_DEBUG(LOG_TAG, "%s Callback thread exited", __func__);
}

 

這個函數作的事情,就像其名字同樣,收到expire的信號以後,作一個dispatch的動做,咱們接下來看看放置到隊列以後如何處理的.

這裏咱們還要看一下當時隊列和線程綁定的狀況:

void alarm_register_processing_queue(fixed_queue_t* queue, thread_t* thread) {
  CHECK(queue != NULL);
  CHECK(thread != NULL);

  fixed_queue_register_dequeue(queue, thread_get_reactor(thread),
                               alarm_queue_ready, NULL);
}

 

咱們看看alarm_queue_ready:

static void alarm_queue_ready(fixed_queue_t* queue, UNUSED_ATTR void* context) {
  CHECK(queue != NULL);

  std::unique_lock<std::mutex> lock(alarms_mutex);
  alarm_t* alarm = (alarm_t*)fixed_queue_try_dequeue(queue);
  if (alarm == NULL) {
    return;  // The alarm was probably canceled
  }

  //
  // If the alarm is not periodic, we've fully serviced it now, and can reset
  // some of its internal state. This is useful to distinguish between expired
  // alarms and active ones.
  //
  alarm_callback_t callback = alarm->callback;
  void* data = alarm->data;
  period_ms_t deadline = alarm->deadline;
  if (alarm->is_periodic) {
    // The periodic alarm has been rescheduled and alarm->deadline has been
    // updated, hence we need to use the previous deadline.
    deadline = alarm->prev_deadline;
  } else {
    alarm->deadline = 0;
    alarm->callback = NULL;
    alarm->data = NULL;
    alarm->queue = NULL;
  }

  std::lock_guard<std::recursive_mutex> cb_lock(*alarm->callback_mutex);
  lock.unlock();

  period_ms_t t0 = now();
  callback(data);//執行callback 函數,並reset alarm
  period_ms_t t1 = now();

  // Update the statistics
  CHECK(t1 >= t0);
  period_ms_t delta = t1 - t0;
  update_scheduling_stats(&alarm->stats, t0, deadline, delta);
}

 

上面流程的核心 就是 取出隊列中的alarm,並執行其中的callback,也就是咱們開定時器的時候的回調函數.

 關於定時器的介紹,就到這裏.

相關文章
相關標籤/搜索