2023-04-17 00:28:05 +08:00
|
|
|
//////////////////////////////////////////////////////////////////////////////////////////
|
|
|
|
// A multi-platform support c++11 library with focus on asynchronous socket I/O for any
|
|
|
|
// client application.
|
|
|
|
//////////////////////////////////////////////////////////////////////////////////////////
|
|
|
|
//
|
|
|
|
// Copyright (c) 2012-2023 HALX99 (halx99 at live dot com)
|
|
|
|
#pragma once
|
|
|
|
|
|
|
|
#include <vector>
|
|
|
|
#include <chrono>
|
2023-05-14 22:39:05 +08:00
|
|
|
#include "yasio/core/socket.hpp"
|
2023-05-14 17:06:43 +08:00
|
|
|
#include "yasio/core/select_interrupter.hpp"
|
2023-04-17 00:28:05 +08:00
|
|
|
|
|
|
|
namespace yasio
|
|
|
|
{
|
|
|
|
YASIO__NS_INLINE
|
|
|
|
namespace inet
|
|
|
|
{
|
|
|
|
class select_io_watcher {
|
|
|
|
public:
|
|
|
|
select_io_watcher()
|
|
|
|
{
|
|
|
|
FD_ZERO(®istered_events_[read_op]);
|
|
|
|
FD_ZERO(®istered_events_[write_op]);
|
|
|
|
FD_ZERO(®istered_events_[except_op]);
|
2023-05-14 17:06:43 +08:00
|
|
|
this->mod_event(interrupter_.read_descriptor(), socket_event::read, 0);
|
2023-04-17 00:28:05 +08:00
|
|
|
}
|
|
|
|
|
2023-05-14 17:06:43 +08:00
|
|
|
void mod_event(socket_native_type fd, int add_events, int remove_events)
|
2023-04-17 00:28:05 +08:00
|
|
|
{
|
2023-05-14 17:06:43 +08:00
|
|
|
if (add_events)
|
|
|
|
{
|
|
|
|
if (yasio__testbits(add_events, socket_event::read))
|
|
|
|
FD_SET(fd, &(registered_events_[read_op]));
|
2023-04-17 00:28:05 +08:00
|
|
|
|
2023-05-14 17:06:43 +08:00
|
|
|
if (yasio__testbits(add_events, socket_event::write))
|
|
|
|
FD_SET(fd, &(registered_events_[write_op]));
|
2023-04-17 00:28:05 +08:00
|
|
|
|
2023-05-14 17:06:43 +08:00
|
|
|
if (yasio__testbits(add_events, socket_event::error))
|
|
|
|
FD_SET(fd, &(registered_events_[except_op]));
|
2023-04-17 00:28:05 +08:00
|
|
|
|
2023-05-14 17:06:43 +08:00
|
|
|
if (max_descriptor_ < static_cast<int>(fd) + 1)
|
|
|
|
max_descriptor_ = static_cast<int>(fd) + 1;
|
|
|
|
}
|
2023-04-17 00:28:05 +08:00
|
|
|
|
2023-05-14 17:06:43 +08:00
|
|
|
if (remove_events)
|
|
|
|
{
|
|
|
|
if (yasio__testbits(remove_events, socket_event::read))
|
|
|
|
FD_CLR(fd, &(registered_events_[read_op]));
|
2023-04-17 00:28:05 +08:00
|
|
|
|
2023-05-14 17:06:43 +08:00
|
|
|
if (yasio__testbits(remove_events, socket_event::write))
|
|
|
|
FD_CLR(fd, &(registered_events_[write_op]));
|
2023-04-17 00:28:05 +08:00
|
|
|
|
2023-05-14 17:06:43 +08:00
|
|
|
if (yasio__testbits(remove_events, socket_event::error))
|
|
|
|
FD_CLR(fd, &(registered_events_[except_op]));
|
|
|
|
}
|
2023-04-17 00:28:05 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
int poll_io(int64_t waitd_us)
|
|
|
|
{
|
|
|
|
::memcpy(this->ready_events_, registered_events_, sizeof(ready_events_));
|
|
|
|
timeval timeout = {(decltype(timeval::tv_sec))(waitd_us / std::micro::den), (decltype(timeval::tv_usec))(waitd_us % std::micro::den)};
|
|
|
|
int num_events = ::select(this->max_descriptor_, &(ready_events_[read_op]), &(ready_events_[write_op]), nullptr, &timeout);
|
|
|
|
if (num_events > 0 && is_ready(this->interrupter_.read_descriptor(), socket_event::read))
|
|
|
|
{
|
|
|
|
if (!interrupter_.reset())
|
|
|
|
interrupter_.recreate();
|
|
|
|
--num_events;
|
|
|
|
}
|
|
|
|
return num_events;
|
|
|
|
}
|
|
|
|
|
|
|
|
void wakeup() { interrupter_.interrupt(); }
|
|
|
|
|
|
|
|
int is_ready(socket_native_type fd, int events) const
|
|
|
|
{
|
|
|
|
int retval = 0;
|
|
|
|
if (events & socket_event::read)
|
|
|
|
retval |= FD_ISSET(fd, &ready_events_[read_op]);
|
|
|
|
if (events & socket_event::write)
|
|
|
|
retval |= FD_ISSET(fd, &ready_events_[write_op]);
|
|
|
|
if (events & socket_event::error)
|
|
|
|
retval |= FD_ISSET(fd, &ready_events_[except_op]);
|
|
|
|
return retval;
|
|
|
|
}
|
|
|
|
|
|
|
|
int max_descriptor() const { return max_descriptor_; }
|
|
|
|
|
|
|
|
protected:
|
|
|
|
enum
|
|
|
|
{
|
|
|
|
read_op,
|
|
|
|
write_op,
|
|
|
|
except_op,
|
|
|
|
max_ops,
|
|
|
|
};
|
|
|
|
fd_set registered_events_[max_ops];
|
|
|
|
fd_set ready_events_[max_ops];
|
|
|
|
int max_descriptor_ = 0;
|
|
|
|
|
|
|
|
select_interrupter interrupter_;
|
|
|
|
};
|
|
|
|
} // namespace inet
|
|
|
|
} // namespace yasio
|