acl/lib_fiber/samples/mbox/main.cpp

149 lines
2.4 KiB
C++

#include "stdafx.h"
#include <stdio.h>
#include <stdlib.h>
class myobj
{
public:
myobj(void) : n_(0) {}
~myobj(void) {}
void run(void)
{
int i = 5;
printf("thread-%lu sleep %d seconds\r\n",
acl::thread::thread_self(), i);
sleep(i);
printf("thread-%lu wakeup\r\n", acl::thread::thread_self());
n_ ++;
}
int get_result(void) const
{
return n_;
}
private:
int n_;
};
class mythread : public acl::thread
{
public:
mythread(acl::mbox<myobj>& mb, myobj& o) : mb_(mb), o_(o) {}
~mythread(void) {}
protected:
// @override
void* run(void)
{
o_.run();
mb_.push(&o_);
return NULL;
}
private:
acl::mbox<myobj>& mb_;
myobj& o_;
};
class myfiber : public acl::fiber
{
public:
myfiber(acl::fiber_sem& sem) : sem_(sem) {}
~myfiber(void) {}
protected:
// @override
void run(void)
{
printf("fiber-%u: wait result from thread\r\n", get_id());
myobj mo;
mythread thr(mb_, mo);
thr.start();
myobj* o = mb_.pop();
assert(o == &mo);
printf("fiber-%u: result = %d\r\n", get_id(), o->get_result());
sem_.post();
delete this;
}
private:
acl::mbox<myobj> mb_;
acl::fiber_sem& sem_;
};
//////////////////////////////////////////////////////////////////////////////
class sleepy_fiber : public acl::fiber
{
public:
sleepy_fiber(acl::fiber_sem& sem) : sem_(sem) {}
~sleepy_fiber(void) {}
protected:
void run(void)
{
time_t begin = time(NULL);
int n = 0;
while (n++ < 6)
{
sleep(1);
printf("fiber-%u sleep %d second\r\n", get_id(), n);
}
time_t end = time(NULL);
printf("fiber-%u wait sem, spent: %ld\r\n",
get_id(), (long) (end - begin));
sem_.wait();
acl::fiber::schedule_stop();
}
private:
acl::fiber_sem& sem_;
};
//////////////////////////////////////////////////////////////////////////////
static void usage(const char* procname)
{
printf("usage: %s -h [help]\r\n", procname);
}
int main(int argc, char *argv[])
{
int ch;
acl::acl_cpp_init();
acl::log::stdout_open(true);
while ((ch = getopt(argc, argv, "h")) > 0)
{
switch (ch)
{
case 'h':
usage(argv[0]);
return 0;
default:
break;
}
}
acl::fiber_sem sem(0);
acl::fiber* f = new myfiber(sem);
f->start();
sleepy_fiber fb(sem);
fb.start();
acl::fiber::schedule();
return 0;
}