0
0
mirror of https://github.com/mongodb/mongo.git synced 2024-11-30 00:56:44 +01:00
mongodb/util/concurrency/task.cpp
2011-01-04 00:40:41 -05:00

181 lines
4.6 KiB
C++

// @file task.cpp
/**
* Copyright (C) 2008 10gen Inc.
*
* This program is free software: you can redistribute it and/or modify
* it under the terms of the GNU Affero General Public License, version 3,
* as published by the Free Software Foundation.
*
* This program is distributed in the hope that it will be useful,b
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU Affero General Public License for more details.
*
* You should have received a copy of the GNU Affero General Public License
* along with this program. If not, see <http://www.gnu.org/licenses/>.
*/
#include "pch.h"
#include <boost/thread/condition.hpp>
#include "task.h"
#include "../goodies.h"
#include "../unittest.h"
#include "../time_support.h"
namespace mongo {
namespace task {
/*void foo() {
boost::mutex m;
boost::mutex::scoped_lock lk(m);
boost::condition cond;
cond.wait(lk);
cond.notify_one();
}*/
Task::Task()
: BackgroundJob( true /* deleteSelf */ ) {
n = 0;
repeat = 0;
}
void Task::halt() { repeat = 0; }
void Task::run() {
assert( n == 0 );
while( 1 ) {
n++;
try {
doWork();
}
catch(...) { }
if( repeat == 0 )
break;
sleepmillis(repeat);
if( inShutdown() )
break;
}
}
void Task::begin() {
go();
}
void fork(Task *t) {
t->begin();
}
void repeat(Task *t, unsigned millis) {
t->repeat = millis;
t->begin();
}
}
}
#include "msg.h"
/* task::Server */
namespace mongo {
namespace task {
/* to get back a return value */
struct Ret {
Ret() : done(false) { }
bool done;
boost::mutex m;
boost::condition c;
const lam *msg;
void f() {
(*msg)();
done = true;
c.notify_one();
}
};
void Server::call( const lam& msg ) {
Ret r;
r.msg = &msg;
lam f = boost::bind(&Ret::f, &r);
send(f);
{
boost::mutex::scoped_lock lk(r.m);
while( !r.done )
r.c.wait(lk);
}
}
void Server::send( lam msg ) {
{
boost::mutex::scoped_lock lk(m);
d.push_back(msg);
}
c.notify_one();
}
void Server::doWork() {
starting();
while( 1 ) {
lam f;
try {
boost::mutex::scoped_lock lk(m);
while( d.empty() )
c.wait(lk);
f = d.front();
d.pop_front();
}
catch(...) {
log() << "ERROR exception in Server:doWork?" << endl;
}
try {
f();
if( rq ) {
rq = false;
{
boost::mutex::scoped_lock lk(m);
d.push_back(f);
}
}
}
catch(std::exception& e) {
log() << "Server::doWork task:" << name() << " exception:" << e.what() << endl;
}
catch(const char *p) {
log() << "Server::doWork task:" << name() << " unknown c exception:" <<
((p&&strlen(p)<800)?p:"?") << endl;
}
catch(...) {
log() << "Server::doWork unknown exception task:" << name() << endl;
}
}
}
static Server *s;
static void abc(int i) {
cout << "Hello " << i << endl;
s->requeue();
}
class TaskUnitTest : public mongo::UnitTest {
public:
virtual void run() {
lam f = boost::bind(abc, 3);
//f();
s = new Server("unittest");
fork(s);
s->send(f);
sleepsecs(30);
cout <<" done" << endl;
}
}; // not running. taskunittest;
}
}