#include "stdafx.h" #include #include static int __delay = 1; ////////////////////////////////////////////////////////////////////////////// class fiber_notifier : public acl::fiber { public: fiber_notifier(acl::wait_group& sync) : sync_(sync) {} private: ~fiber_notifier(void) {} private: acl::wait_group& sync_; // @override void run(void) { printf("Fiber-%u started, delay=%d seconds\r\n", acl::fiber::self(), __delay); sleep(__delay); printf("Fiber-%u done\r\n", acl::fiber::self()); sync_.done(); delete this; } }; ////////////////////////////////////////////////////////////////////////////// class thread_notifier : public acl::thread { public: thread_notifier(acl::wait_group& sync) : sync_(sync) {} private: ~thread_notifier(void) {} private: acl::wait_group& sync_; // @override void* run(void) { printf("Thread-%lu started, delay=%d seconds\r\n", acl::thread::self(), __delay); sleep(__delay); printf("Thread-%lu done\r\n", acl::thread::self()); sync_.done(); delete this; return NULL; } }; ////////////////////////////////////////////////////////////////////////////// class fiber_waiter : public acl::fiber { public: fiber_waiter(acl::wait_group& sync) : sync_(sync) {} ~fiber_waiter(void) {} private: acl::wait_group& sync_; // @override void run(void) { size_t ret = sync_.wait(); printf("All threads and fibers were done, ret=%zd\r\n", ret); } }; ////////////////////////////////////////////////////////////////////////////// static void usage(const char* procname) { printf("usage: %s -h [help]\r\n" " -e event_type[kernel|poll|select|io_uring]\r\n" " -t threads_count[default: 1]\r\n" " -c fibers_count[default: 1]\r\n" " -d delay[default: 1 second]\r\n" , procname); } int main(int argc, char *argv[]) { int ch, threads_count = 1, fibers_count = 1; acl::fiber_event_t event_type = acl::FIBER_EVENT_T_KERNEL; acl::acl_cpp_init(); acl::log::stdout_open(true); while ((ch = getopt(argc, argv, "he:t:f:c:d:")) > 0) { switch (ch) { case 'h': usage(argv[0]); return 0; case 'e': if (strcasecmp(optarg, "kernel") == 0) { event_type = acl::FIBER_EVENT_T_KERNEL; } else if (strcasecmp(optarg, "select") == 0) { event_type = acl::FIBER_EVENT_T_SELECT; } else if (strcasecmp(optarg, "poll") == 0) { event_type = acl::FIBER_EVENT_T_POLL; } else if (strcasecmp(optarg, "io_uring") == 0) { event_type = acl::FIBER_EVENT_T_IO_URING; } break; case 't': threads_count = atoi(optarg); break; case 'c': fibers_count = atoi(optarg); break; case 'd': __delay = atoi(optarg); break; default: break; } } acl::log::stdout_open(true); acl::fiber::stdout_open(true); acl::wait_group sync; sync.add(threads_count + fibers_count); for (int i = 0; i < threads_count; i++) { acl::thread* thr = new thread_notifier(sync); thr->set_detachable(true); thr->start(); } for (int i = 0; i < fibers_count; i++) { acl::fiber* fb = new fiber_notifier(sync); fb->start(); } fiber_waiter waiter(sync); waiter.start(); acl::fiber::schedule_with(event_type); return 0; }