| Index: base/mac/libdispatch_task_runner.cc | 
| diff --git a/base/mac/libdispatch_task_runner.cc b/base/mac/libdispatch_task_runner.cc | 
| new file mode 100644 | 
| index 0000000000000000000000000000000000000000..0c1eebc5c5e7b606abaad876edf0e8ff3cb579b4 | 
| --- /dev/null | 
| +++ b/base/mac/libdispatch_task_runner.cc | 
| @@ -0,0 +1,83 @@ | 
| +// Copyright (c) 2012 The Chromium Authors. All rights reserved. | 
| +// Use of this source code is governed by a BSD-style license that can be | 
| +// found in the LICENSE file. | 
| + | 
| +#include "base/mac/libdispatch_task_runner.h" | 
| + | 
| +#include <stdint.h> | 
| + | 
| +#include "base/callback.h" | 
| + | 
| +namespace base { | 
| +namespace mac { | 
| + | 
| +LibDispatchTaskRunner::LibDispatchTaskRunner(const char* name) | 
| +    : queue_(dispatch_queue_create(name, NULL)), | 
| +      queue_finalized_(base::WaitableEvent::ResetPolicy::AUTOMATIC, | 
| +                       base::WaitableEvent::InitialState::NOT_SIGNALED) { | 
| +  dispatch_set_context(queue_, this); | 
| +  dispatch_set_finalizer_f(queue_, &LibDispatchTaskRunner::Finalizer); | 
| +} | 
| + | 
| +bool LibDispatchTaskRunner::PostDelayedTask( | 
| +    const tracked_objects::Location& from_here, | 
| +    OnceClosure task, | 
| +    base::TimeDelta delay) { | 
| +  if (!queue_) | 
| +    return false; | 
| + | 
| +  // The block runtime would implicitly copy the reference, not the object | 
| +  // it's referencing. Copy the closure into block storage so it's available | 
| +  // to run. | 
| +  __block OnceClosure task_copy = std::move(task); | 
| +  void(^run_task)(void) = ^{ | 
| +      std::move(task_copy).Run(); | 
| +  }; | 
| + | 
| +  int64_t delay_nano = | 
| +      delay.InMicroseconds() * base::Time::kNanosecondsPerMicrosecond; | 
| +  if (delay_nano > 0) { | 
| +    dispatch_time_t time = dispatch_time(DISPATCH_TIME_NOW, delay_nano); | 
| +    dispatch_after(time, queue_, run_task); | 
| +  } else { | 
| +    dispatch_async(queue_, run_task); | 
| +  } | 
| +  return true; | 
| +} | 
| + | 
| +bool LibDispatchTaskRunner::RunsTasksOnCurrentThread() const { | 
| +  return queue_ == dispatch_get_current_queue(); | 
| +} | 
| + | 
| +bool LibDispatchTaskRunner::PostNonNestableDelayedTask( | 
| +    const tracked_objects::Location& from_here, | 
| +    OnceClosure task, | 
| +    base::TimeDelta delay) { | 
| +  return PostDelayedTask(from_here, std::move(task), delay); | 
| +} | 
| + | 
| +void LibDispatchTaskRunner::Shutdown() { | 
| +  dispatch_release(queue_); | 
| +  queue_ = NULL; | 
| +  queue_finalized_.Wait(); | 
| +} | 
| + | 
| +dispatch_queue_t LibDispatchTaskRunner::GetDispatchQueue() const { | 
| +  return queue_; | 
| +} | 
| + | 
| +LibDispatchTaskRunner::~LibDispatchTaskRunner() { | 
| +  if (queue_) { | 
| +    dispatch_set_context(queue_, NULL); | 
| +    dispatch_set_finalizer_f(queue_, NULL); | 
| +    dispatch_release(queue_); | 
| +  } | 
| +} | 
| + | 
| +void LibDispatchTaskRunner::Finalizer(void* context) { | 
| +  LibDispatchTaskRunner* self = static_cast<LibDispatchTaskRunner*>(context); | 
| +  self->queue_finalized_.Signal(); | 
| +} | 
| + | 
| +}  // namespace mac | 
| +}  // namespace base | 
|  |