私はepollに基づいた単純なサーバークラスを書いています。目を覚ますためにepoll_wait()
、eventfdを使うことにしました。シンプルなイベントコミュニケーションに向いているとのことで、そう思います。だから私は自分のイベントを作成し、それに時計を置きました:
_epollfd = epoll_create1(0);
if (_epollfd == -1) throw ServerError("epoll_create");
_eventfd = eventfd(0, EFD_NONBLOCK);
epoll_event evnt = {0};
evnt.data.fd = _eventfd;
evnt.events = _events;
if (epoll_ctl(_epollfd, EPOLL_CTL_ADD, _eventfd, &evnt) == -1)
throw ServerError("epoll_ctl(add)");
メッセージ待機ループの後半、別のスレッドで:
int count = epoll_wait(_epollfd, evnts, EVENTS, -1);
if (count == -1)
{
if (errno != EINTR)
{
perror("epoll_wait");
return;
}
}
for (int i = 0; i < count; ++i)
{
epoll_event & e = evnts[i];
if (e.data.fd == _serverSock)
connectionAccepted();
else if (e.data.fd == _eventfd)
{
eventfd_t val;
eventfd_read(_eventfd, &val);
return;
}
}
そしてもちろん、サーバーを停止するコードは次のとおりです。
eventfd_write(_eventfd, 1);
epoll_wait()
説明できない理由で、イベントに書いただけでは目を覚ますことができませんでした。最終的に、これはいくつかのデバッグセッションで機能しました。
これが私の回避策ですEPOLLOUT
。fdが書き込み可能になるたびにイベントがトリガーされることを知っているので、停止コードを次のように変更しました。
epoll_event evnt = {0};
evnt.data.fd = _eventfd;
evnt.events = EPOLLOUT;
if (epoll_ctl(_epollfd, EPOLL_CTL_MOD, _eventfd, &evnt) == -1)
throw ServerError("epoll_ctl(mod)");
今は動作しますが、このようにすべきではありません。
これは難しいことではないと思います。私は何を間違えましたか?
ありがとう