2009-10-27 20:58:27 +01:00
|
|
|
// background.h
|
|
|
|
|
|
|
|
/* Copyright 2009 10gen Inc.
|
|
|
|
*
|
|
|
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
* you may not use this file except in compliance with the License.
|
|
|
|
* You may obtain a copy of the License at
|
|
|
|
*
|
|
|
|
* http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
*
|
|
|
|
* Unless required by applicable law or agreed to in writing, software
|
|
|
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
* See the License for the specific language governing permissions and
|
|
|
|
* limitations under the License.
|
|
|
|
*/
|
2008-10-29 22:48:03 +01:00
|
|
|
|
|
|
|
#pragma once
|
|
|
|
|
2009-01-14 23:09:51 +01:00
|
|
|
namespace mongo {
|
|
|
|
|
2010-04-18 18:30:40 +02:00
|
|
|
/** object-orienty background thread dispatching.
|
2008-10-29 22:48:03 +01:00
|
|
|
|
2009-01-15 16:17:11 +01:00
|
|
|
subclass and define run()
|
2008-10-29 22:48:03 +01:00
|
|
|
|
2010-04-18 18:30:40 +02:00
|
|
|
It is ok to call go(), that is, run the job, more than once -- if the
|
|
|
|
previous invocation
|
2009-01-15 16:17:11 +01:00
|
|
|
has finished. Thus one pattern of use is to embed a backgroundjob
|
|
|
|
in your object and reuse it (or same thing with inheritance).
|
2010-04-18 18:30:40 +02:00
|
|
|
|
|
|
|
note when job destructs, the thread is not terminated if still running.
|
|
|
|
generally if the thread could still be running, allocate the job dynamically
|
|
|
|
and set deleteSelf to true.
|
|
|
|
*/
|
|
|
|
/* example
|
|
|
|
class ConnectBG : public BackgroundJob {
|
|
|
|
public:
|
|
|
|
int sock;
|
|
|
|
int res;
|
|
|
|
SockAddr farEnd;
|
|
|
|
void run() {
|
|
|
|
res = ::connect(sock, farEnd.raw(), farEnd.addressSize);
|
|
|
|
}
|
|
|
|
};
|
2009-01-15 16:17:11 +01:00
|
|
|
*/
|
2010-04-18 18:30:40 +02:00
|
|
|
class BackgroundJob : boost::noncopyable {
|
2009-01-15 16:17:11 +01:00
|
|
|
protected:
|
2010-04-18 18:30:40 +02:00
|
|
|
/** define this to do your work!
|
|
|
|
after this returns, state is set to done
|
|
|
|
(and deletes if deleteSelf true)
|
|
|
|
*/
|
2009-01-15 16:17:11 +01:00
|
|
|
virtual void run() = 0;
|
2008-10-29 22:48:03 +01:00
|
|
|
|
2009-01-15 16:17:11 +01:00
|
|
|
public:
|
|
|
|
enum State {
|
|
|
|
NotStarted,
|
|
|
|
Running,
|
|
|
|
Done
|
|
|
|
};
|
|
|
|
State getState() const {
|
|
|
|
return state;
|
|
|
|
}
|
|
|
|
bool running() const {
|
|
|
|
return state == Running;
|
|
|
|
}
|
2008-10-29 22:48:03 +01:00
|
|
|
|
2009-01-15 16:17:11 +01:00
|
|
|
bool deleteSelf; // delete self when Done?
|
2008-10-29 22:48:03 +01:00
|
|
|
|
2009-01-15 16:17:11 +01:00
|
|
|
BackgroundJob() {
|
|
|
|
deleteSelf = false;
|
|
|
|
state = NotStarted;
|
|
|
|
}
|
|
|
|
virtual ~BackgroundJob() { }
|
2008-10-29 22:48:03 +01:00
|
|
|
|
2009-01-15 16:17:11 +01:00
|
|
|
// start job. returns before it's finished.
|
|
|
|
BackgroundJob& go();
|
2008-10-29 22:48:03 +01:00
|
|
|
|
2009-01-15 16:17:11 +01:00
|
|
|
// wait for completion. this spins with sleep() so not terribly efficient.
|
|
|
|
// returns true if did not time out.
|
|
|
|
//
|
|
|
|
// note you can call wait() more than once if the first call times out.
|
|
|
|
bool wait(int msMax = 0);
|
2008-10-29 22:48:03 +01:00
|
|
|
|
2009-01-15 16:17:11 +01:00
|
|
|
private:
|
|
|
|
static BackgroundJob *grab;
|
2010-03-15 17:42:01 +01:00
|
|
|
static mongo::mutex mutex;
|
2009-01-15 16:17:11 +01:00
|
|
|
static void thr();
|
|
|
|
volatile State state;
|
|
|
|
};
|
2009-01-14 23:09:51 +01:00
|
|
|
|
|
|
|
} // namespace mongo
|