Home | History | Annotate | Download | only in quic
      1 // Copyright (c) 2012 The Chromium Authors. All rights reserved.
      2 // Use of this source code is governed by a BSD-style license that can be
      3 // found in the LICENSE file.
      4 
      5 #include "net/quic/quic_connection_helper.h"
      6 
      7 #include "base/location.h"
      8 #include "base/logging.h"
      9 #include "base/metrics/sparse_histogram.h"
     10 #include "base/task_runner.h"
     11 #include "base/time/time.h"
     12 #include "net/base/io_buffer.h"
     13 #include "net/base/net_errors.h"
     14 #include "net/quic/quic_utils.h"
     15 
     16 namespace net {
     17 
     18 namespace {
     19 
     20 class QuicChromeAlarm : public QuicAlarm {
     21  public:
     22   QuicChromeAlarm(const QuicClock* clock,
     23                   base::TaskRunner* task_runner,
     24                   QuicAlarm::Delegate* delegate)
     25       : QuicAlarm(delegate),
     26         clock_(clock),
     27         task_runner_(task_runner),
     28         task_deadline_(QuicTime::Zero()),
     29         weak_factory_(this) {}
     30 
     31  protected:
     32   virtual void SetImpl() OVERRIDE {
     33     DCHECK(deadline().IsInitialized());
     34     if (task_deadline_.IsInitialized()) {
     35       if (task_deadline_ <= deadline()) {
     36         // Since tasks can not be un-posted, OnAlarm will be invoked which
     37         // will notice that deadline has not yet been reached, and will set
     38         // the alarm for the new deadline.
     39         return;
     40       }
     41       // The scheduled task is after new deadline.  Invalidate the weak ptrs
     42       // so that task does not execute when we're not expecting it.
     43       weak_factory_.InvalidateWeakPtrs();
     44     }
     45 
     46     int64 delay_us = deadline().Subtract(clock_->Now()).ToMicroseconds();
     47     if (delay_us < 0) {
     48       delay_us = 0;
     49     }
     50     task_runner_->PostDelayedTask(
     51         FROM_HERE,
     52         base::Bind(&QuicChromeAlarm::OnAlarm, weak_factory_.GetWeakPtr()),
     53         base::TimeDelta::FromMicroseconds(delay_us));
     54     task_deadline_ = deadline();
     55   }
     56 
     57   virtual void CancelImpl() OVERRIDE {
     58     DCHECK(!deadline().IsInitialized());
     59     // Since tasks can not be un-posted, OnAlarm will be invoked which
     60     // will notice that deadline is not Initialized and will do nothing.
     61   }
     62 
     63  private:
     64   void OnAlarm() {
     65     DCHECK(task_deadline_.IsInitialized());
     66     task_deadline_ = QuicTime::Zero();
     67     // The alarm may have been cancelled.
     68     if (!deadline().IsInitialized()) {
     69       return;
     70     }
     71 
     72     // The alarm may have been re-set to a later time.
     73     if (clock_->Now() < deadline()) {
     74       SetImpl();
     75       return;
     76     }
     77 
     78     Fire();
     79   }
     80 
     81   const QuicClock* clock_;
     82   base::TaskRunner* task_runner_;
     83   // If a task has been posted to the message loop, this is the time it
     84   // was scheduled to fire.  Tracking this allows us to avoid posting a
     85   // new tast if the new deadline is in the future, but permits us to
     86   // post a new task when the new deadline now earlier than when
     87   // previously posted.
     88   QuicTime task_deadline_;
     89   base::WeakPtrFactory<QuicChromeAlarm> weak_factory_;
     90 };
     91 
     92 }  // namespace
     93 
     94 QuicConnectionHelper::QuicConnectionHelper(base::TaskRunner* task_runner,
     95                                            const QuicClock* clock,
     96                                            QuicRandom* random_generator)
     97     : weak_factory_(this),
     98       task_runner_(task_runner),
     99       clock_(clock),
    100       random_generator_(random_generator) {
    101 }
    102 
    103 QuicConnectionHelper::~QuicConnectionHelper() {
    104 }
    105 
    106 const QuicClock* QuicConnectionHelper::GetClock() const {
    107   return clock_;
    108 }
    109 
    110 QuicRandom* QuicConnectionHelper::GetRandomGenerator() {
    111   return random_generator_;
    112 }
    113 
    114 QuicAlarm* QuicConnectionHelper::CreateAlarm(QuicAlarm::Delegate* delegate) {
    115   return new QuicChromeAlarm(clock_, task_runner_, delegate);
    116 }
    117 
    118 }  // namespace net
    119