12745a763SShuo Chen// excerpts from http://code.google.com/p/muduo/
22745a763SShuo Chen//
32745a763SShuo Chen// Use of this source code is governed by a BSD-style license
42745a763SShuo Chen// that can be found in the License file.
52745a763SShuo Chen//
62745a763SShuo Chen// Author: Shuo Chen (chenshuo at chenshuo dot com)
72745a763SShuo Chen
82745a763SShuo Chen#include "Poller.h"
92745a763SShuo Chen
102745a763SShuo Chen#include "Channel.h"
112745a763SShuo Chen#include "logging/Logging.h"
122745a763SShuo Chen
132745a763SShuo Chen#include <assert.h>
142745a763SShuo Chen#include <poll.h>
152745a763SShuo Chen
162745a763SShuo Chenusing namespace muduo;
172745a763SShuo Chen
182745a763SShuo ChenPoller::Poller(EventLoop* loop)
192745a763SShuo Chen  : ownerLoop_(loop)
202745a763SShuo Chen{
212745a763SShuo Chen}
222745a763SShuo Chen
232745a763SShuo ChenPoller::~Poller()
242745a763SShuo Chen{
252745a763SShuo Chen}
262745a763SShuo Chen
272745a763SShuo ChenTimestamp Poller::poll(int timeoutMs, ChannelList* activeChannels)
282745a763SShuo Chen{
292745a763SShuo Chen  // XXX pollfds_ shouldn't change
302745a763SShuo Chen  int numEvents = ::poll(&*pollfds_.begin(), pollfds_.size(), timeoutMs);
312745a763SShuo Chen  Timestamp now(Timestamp::now());
32b4a5ce52SShuo Chen  if (numEvents > 0) {
332745a763SShuo Chen    LOG_TRACE << numEvents << " events happended";
342745a763SShuo Chen    fillActiveChannels(numEvents, activeChannels);
35b4a5ce52SShuo Chen  } else if (numEvents == 0) {
362745a763SShuo Chen    LOG_TRACE << " nothing happended";
37b4a5ce52SShuo Chen  } else {
382745a763SShuo Chen    LOG_SYSERR << "Poller::poll()";
392745a763SShuo Chen  }
402745a763SShuo Chen  return now;
412745a763SShuo Chen}
422745a763SShuo Chen
432745a763SShuo Chenvoid Poller::fillActiveChannels(int numEvents,
442745a763SShuo Chen                                ChannelList* activeChannels) const
452745a763SShuo Chen{
462745a763SShuo Chen  for (PollFdList::const_iterator pfd = pollfds_.begin();
472745a763SShuo Chen      pfd != pollfds_.end() && numEvents > 0; ++pfd)
482745a763SShuo Chen  {
492745a763SShuo Chen    if (pfd->revents > 0)
502745a763SShuo Chen    {
512745a763SShuo Chen      --numEvents;
522745a763SShuo Chen      ChannelMap::const_iterator ch = channels_.find(pfd->fd);
532745a763SShuo Chen      assert(ch != channels_.end());
542745a763SShuo Chen      Channel* channel = ch->second;
552745a763SShuo Chen      assert(channel->fd() == pfd->fd);
562745a763SShuo Chen      channel->set_revents(pfd->revents);
572745a763SShuo Chen      // pfd->revents = 0;
582745a763SShuo Chen      activeChannels->push_back(channel);
592745a763SShuo Chen    }
602745a763SShuo Chen  }
612745a763SShuo Chen}
622745a763SShuo Chen
632745a763SShuo Chenvoid Poller::updateChannel(Channel* channel)
642745a763SShuo Chen{
65ee021d99SShuo Chen  assertInLoopThread();
662745a763SShuo Chen  LOG_TRACE << "fd = " << channel->fd() << " events = " << channel->events();
67b4a5ce52SShuo Chen  if (channel->index() < 0) {
682745a763SShuo Chen    // a new one, add to pollfds_
692745a763SShuo Chen    assert(channels_.find(channel->fd()) == channels_.end());
702745a763SShuo Chen    struct pollfd pfd;
712745a763SShuo Chen    pfd.fd = channel->fd();
722745a763SShuo Chen    pfd.events = static_cast<short>(channel->events());
732745a763SShuo Chen    pfd.revents = 0;
742745a763SShuo Chen    pollfds_.push_back(pfd);
752745a763SShuo Chen    int idx = static_cast<int>(pollfds_.size())-1;
762745a763SShuo Chen    channel->set_index(idx);
772745a763SShuo Chen    channels_[pfd.fd] = channel;
78b4a5ce52SShuo Chen  } else {
792745a763SShuo Chen    // update existing one
802745a763SShuo Chen    assert(channels_.find(channel->fd()) != channels_.end());
812745a763SShuo Chen    assert(channels_[channel->fd()] == channel);
822745a763SShuo Chen    int idx = channel->index();
832745a763SShuo Chen    assert(0 <= idx && idx < static_cast<int>(pollfds_.size()));
842745a763SShuo Chen    struct pollfd& pfd = pollfds_[idx];
852745a763SShuo Chen    assert(pfd.fd == channel->fd() || pfd.fd == -1);
8617c057c3SShuo Chen    pfd.events = static_cast<short>(channel->events());
8717c057c3SShuo Chen    pfd.revents = 0;
88b4a5ce52SShuo Chen    if (channel->isNoneEvent()) {
892745a763SShuo Chen      // ignore this pollfd
902745a763SShuo Chen      pfd.fd = -1;
912745a763SShuo Chen    }
922745a763SShuo Chen  }
932745a763SShuo Chen}
942745a763SShuo Chen
95