1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
|
// 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 "chrome/browser/sync/glue/history_model_worker.h"
#include "base/memory/ref_counted.h"
#include "base/message_loop.h"
#include "base/synchronization/waitable_event.h"
#include "content/public/browser/browser_thread.h"
using base::WaitableEvent;
using content::BrowserThread;
namespace browser_sync {
class WorkerTask : public history::HistoryDBTask {
public:
WorkerTask(
const syncer::WorkCallback& work,
WaitableEvent* done,
syncer::SyncerError* error)
: work_(work), done_(done), error_(error) {}
virtual bool RunOnDBThread(history::HistoryBackend* backend,
history::HistoryDatabase* db) OVERRIDE {
*error_ = work_.Run();
done_->Signal();
return true;
}
// Since the DoWorkAndWaitUntilDone() is synchronous, we don't need to run
// any code asynchronously on the main thread after completion.
virtual void DoneRunOnMainThread() OVERRIDE {}
protected:
virtual ~WorkerTask() {}
syncer::WorkCallback work_;
WaitableEvent* done_;
syncer::SyncerError* error_;
};
class AddDBThreadObserverTask : public history::HistoryDBTask {
public:
AddDBThreadObserverTask(HistoryModelWorker* history_worker)
: history_worker_(history_worker) {}
virtual bool RunOnDBThread(history::HistoryBackend* backend,
history::HistoryDatabase* db) OVERRIDE {
MessageLoop::current()->AddDestructionObserver(history_worker_.get());
return true;
}
virtual void DoneRunOnMainThread() OVERRIDE {}
private:
virtual ~AddDBThreadObserverTask() {}
scoped_refptr<HistoryModelWorker> history_worker_;
};
namespace {
// Post the work task on |history_service|'s DB thread from the UI
// thread.
void PostWorkerTask(const base::WeakPtr<HistoryService>& history_service,
const syncer::WorkCallback& work,
CancelableRequestConsumerT<int, 0>* cancelable_consumer,
WaitableEvent* done,
syncer::SyncerError* error) {
DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
if (history_service) {
scoped_refptr<WorkerTask> task(new WorkerTask(work, done, error));
history_service->ScheduleDBTask(task.get(), cancelable_consumer);
} else {
*error = syncer::CANNOT_DO_WORK;
done->Signal();
}
}
} // namespace
HistoryModelWorker::HistoryModelWorker(
const base::WeakPtr<HistoryService>& history_service,
syncer::WorkerLoopDestructionObserver* observer)
: syncer::ModelSafeWorker(observer),
history_service_(history_service) {
CHECK(history_service.get());
}
void HistoryModelWorker::RegisterForLoopDestruction() {
CHECK(history_service_.get());
history_service_->ScheduleDBTask(new AddDBThreadObserverTask(this),
&cancelable_consumer_);
}
syncer::SyncerError HistoryModelWorker::DoWorkAndWaitUntilDoneImpl(
const syncer::WorkCallback& work) {
syncer::SyncerError error = syncer::UNSET;
if (BrowserThread::PostTask(BrowserThread::UI, FROM_HERE,
base::Bind(&PostWorkerTask, history_service_,
work, &cancelable_consumer_,
work_done_or_stopped(),
&error))) {
work_done_or_stopped()->Wait();
} else {
error = syncer::CANNOT_DO_WORK;
}
return error;
}
syncer::ModelSafeGroup HistoryModelWorker::GetModelSafeGroup() {
return syncer::GROUP_HISTORY;
}
HistoryModelWorker::~HistoryModelWorker() {}
} // namespace browser_sync
|