Chromium Code Reviews| Index: base/sequenced_task_runner_helpers.h |
| diff --git a/base/sequenced_task_runner_helpers.h b/base/sequenced_task_runner_helpers.h |
| new file mode 100644 |
| index 0000000000000000000000000000000000000000..eb0252f527503584fb8b9b753f683edd0a6b0b44 |
| --- /dev/null |
| +++ b/base/sequenced_task_runner_helpers.h |
| @@ -0,0 +1,113 @@ |
| +// 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. |
| + |
| +#ifndef BASE_SEQUENCED_TASK_RUNNER_HELPERS_H_ |
| +#define BASE_SEQUENCED_TASK_RUNNER_HELPERS_H_ |
| +#pragma once |
| + |
| +#include "base/basictypes.h" |
| + |
| +// TODO(akalin): Investigate whether it's possible to just have |
| +// SequencedTaskRunner use these helpers (instead of MessageLoop). |
| +// Then we can just move these to sequenced_task_runner.h. |
| + |
| +namespace tracked_objects { |
| +class Location; |
| +} |
| + |
| +namespace base { |
| + |
| +namespace subtle { |
| +template <class T, class R> class DeleteHelperInternal; |
| +template <class T, class R> class ReleaseHelperInternal; |
| +} |
| + |
| +// Template helpers which use a function indirection to erase T from |
|
willchan no longer on Chromium
2012/02/09 20:50:50
s/a //
akalin
2012/02/10 22:48:59
Done.
|
| +// the function signature while still remembering it so we can call |
| +// the correct destructor/release function. |
| +// |
| +// We use this trick so we don't need to include bind.h in a header |
| +// file like sequenced_task_runner.h. We also wrap the helpers in a |
| +// templated class to make it easier for users of DeleteSoon to |
| +// declare the helper as a friend. |
| +template <class T> |
| +class DeleteHelper { |
| + private: |
|
willchan no longer on Chromium
2012/02/09 20:50:50
I think you're being too draconian here with all t
akalin
2012/02/10 22:48:59
Moving DeleteHelper into internal won't work, beca
|
| + template <class T2, class R> friend class subtle::DeleteHelperInternal; |
| + |
| + static void DoDelete(const void* object) { |
| + delete reinterpret_cast<const T*>(object); |
| + } |
| + |
| + DISALLOW_COPY_AND_ASSIGN(DeleteHelper); |
| +}; |
| + |
| +template <class T> |
| +class ReleaseHelper { |
| + private: |
| + template <class T2, class R> friend class subtle::ReleaseHelperInternal; |
| + |
| + static void DoRelease(const void* object) { |
| + reinterpret_cast<const T*>(object)->Release(); |
| + } |
| + |
| + DISALLOW_COPY_AND_ASSIGN(ReleaseHelper); |
| +}; |
| + |
| +namespace subtle { |
| + |
| +// An internal SequencedTaskRunner-like class helper for DeleteHelper |
| +// and ReleaseHelper. We don't want to expose the Do*() functions |
| +// directly directly since the void* argument makes it possible to |
| +// pass/ an object of the wrong type to delete. Instead, we force |
| +// callers to go through these internal helpers for type |
| +// safety. SequencedTaskRunner-like classes which expose DeleteSoon or |
| +// ReleaseSoon methods should friend the appropriate helper and |
| +// implement a corresponding *Internal method with the following |
| +// signature: |
| +// |
| +// bool(const tracked_objects::Location&, |
| +// void(*function)(const void*), |
| +// void* object) |
| +// |
| +// An implementation of this function should simply create a |
| +// base::Closure from (function, object) and return the result of |
| +// posting the task. |
| +template <class T, class ReturnType> |
| +class DeleteHelperInternal { |
| + public: |
| + template <class SequencedTaskRunnerType> |
| + static ReturnType DeleteViaSequencedTaskRunner( |
| + SequencedTaskRunnerType* sequenced_task_runner, |
| + const tracked_objects::Location& from_here, |
| + const T* object) { |
| + return sequenced_task_runner->DeleteSoonInternal( |
| + from_here, &DeleteHelper<T>::DoDelete, object); |
| + } |
| + |
| + private: |
| + DISALLOW_COPY_AND_ASSIGN(DeleteHelperInternal); |
| +}; |
| + |
| +template <class T, class ReturnType> |
| +class ReleaseHelperInternal { |
| + public: |
| + template <class SequencedTaskRunnerType> |
| + static ReturnType ReleaseViaSequencedTaskRunner( |
| + SequencedTaskRunnerType* sequenced_task_runner, |
| + const tracked_objects::Location& from_here, |
| + const T* object) { |
| + return sequenced_task_runner->ReleaseSoonInternal( |
| + from_here, &ReleaseHelper<T>::DoRelease, object); |
| + } |
| + |
| + private: |
| + DISALLOW_COPY_AND_ASSIGN(ReleaseHelperInternal); |
| +}; |
| + |
| +} // namespace subtle |
| + |
| +} // namespace base |
| + |
| +#endif // BASE_SEQUENCED_TASK_RUNNER_HELPERS_H_ |