| // Copyright (c) 2011 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. |
| |
| // Illustrates how to use worker threads that issue completion callbacks |
| |
| #include "base/bind.h" |
| #include "base/logging.h" |
| #include "base/message_loop.h" |
| #include "base/threading/worker_pool.h" |
| #include "net/base/completion_callback.h" |
| #include "net/base/test_completion_callback.h" |
| #include "testing/gtest/include/gtest/gtest.h" |
| #include "testing/platform_test.h" |
| |
| typedef PlatformTest TestCompletionCallbackTest; |
| |
| const int kMagicResult = 8888; |
| |
| // ExampleEmployer is a toy version of HostResolver |
| // TODO: restore damage done in extracting example from real code |
| // (e.g. bring back real destructor, bring back comments) |
| class ExampleEmployer { |
| public: |
| ExampleEmployer(); |
| ~ExampleEmployer(); |
| |
| // Do some imaginary work on a worker thread; |
| // when done, worker posts callback on the original thread. |
| // Returns true on success |
| bool DoSomething(const net::CompletionCallback& callback); |
| |
| private: |
| class ExampleWorker; |
| friend class ExampleWorker; |
| scoped_refptr<ExampleWorker> request_; |
| DISALLOW_COPY_AND_ASSIGN(ExampleEmployer); |
| }; |
| |
| // Helper class; this is how ExampleEmployer puts work on a different thread |
| class ExampleEmployer::ExampleWorker |
| : public base::RefCountedThreadSafe<ExampleWorker> { |
| public: |
| ExampleWorker(ExampleEmployer* employer, |
| const net::CompletionCallback& callback) |
| : employer_(employer), callback_(callback), |
| origin_loop_(MessageLoop::current()) { |
| } |
| void DoWork(); |
| void DoCallback(); |
| private: |
| friend class base::RefCountedThreadSafe<ExampleWorker>; |
| |
| ~ExampleWorker() {} |
| |
| // Only used on the origin thread (where DoSomething was called). |
| ExampleEmployer* employer_; |
| net::CompletionCallback callback_; |
| // Used to post ourselves onto the origin thread. |
| base::Lock origin_loop_lock_; |
| MessageLoop* origin_loop_; |
| }; |
| |
| void ExampleEmployer::ExampleWorker::DoWork() { |
| // Running on the worker thread |
| // In a real worker thread, some work would be done here. |
| // Pretend it is, and send the completion callback. |
| |
| // The origin loop could go away while we are trying to post to it, so we |
| // need to call its PostTask method inside a lock. See ~ExampleEmployer. |
| { |
| base::AutoLock locked(origin_loop_lock_); |
| if (origin_loop_) |
| origin_loop_->PostTask(FROM_HERE, |
| base::Bind(&ExampleWorker::DoCallback, this)); |
| } |
| } |
| |
| void ExampleEmployer::ExampleWorker::DoCallback() { |
| // Running on the origin thread. |
| |
| // Drop the employer_'s reference to us. Do this before running the |
| // callback since the callback might result in the employer being |
| // destroyed. |
| employer_->request_ = NULL; |
| |
| callback_.Run(kMagicResult); |
| } |
| |
| ExampleEmployer::ExampleEmployer() { |
| } |
| |
| ExampleEmployer::~ExampleEmployer() { |
| } |
| |
| bool ExampleEmployer::DoSomething(const net::CompletionCallback& callback) { |
| DCHECK(!request_) << "already in use"; |
| |
| request_ = new ExampleWorker(this, callback); |
| |
| // Dispatch to worker thread... |
| if (!base::WorkerPool::PostTask( |
| FROM_HERE, |
| base::Bind(&ExampleWorker::DoWork, request_.get()), |
| true)) { |
| NOTREACHED(); |
| request_ = NULL; |
| return false; |
| } |
| |
| return true; |
| } |
| |
| TEST_F(TestCompletionCallbackTest, Simple) { |
| ExampleEmployer boss; |
| net::TestCompletionCallback callback; |
| bool queued = boss.DoSomething(callback.callback()); |
| EXPECT_EQ(queued, true); |
| int result = callback.WaitForResult(); |
| EXPECT_EQ(result, kMagicResult); |
| } |
| |
| // TODO: test deleting ExampleEmployer while work outstanding |