mirror of
https://github.com/fish-shell/fish-shell
synced 2025-01-28 12:45:13 +00:00
Allow using poll() to check for readability
Cygwin tests are failing because cygwin has a low limit of only 64 fds in select(). Extend select_wrapper_t to also support using poll(), according to a FISH_USE_POLL new define. All systems now use poll() except for Mac. Rename select_wrapper_t to fd_readable_set_t since now it may not wrap select(). This allows the deep-cmdsub.fish test to pass on Cygwin.
This commit is contained in:
parent
5e67a299ae
commit
57a9fe492e
10 changed files with 120 additions and 37 deletions
|
@ -1459,7 +1459,7 @@ class universal_notifier_named_pipe_t final : public universal_notifier_t {
|
||||||
// If we're no longer readable, go back to wait mode.
|
// If we're no longer readable, go back to wait mode.
|
||||||
// Conversely, if we have been readable too long, perhaps some fish died while its
|
// Conversely, if we have been readable too long, perhaps some fish died while its
|
||||||
// written data was still on the pipe; drain some.
|
// written data was still on the pipe; drain some.
|
||||||
if (!select_wrapper_t::poll_fd_readable(pipe_fd.fd())) {
|
if (!fd_readable_set_t::poll_fd_readable(pipe_fd.fd())) {
|
||||||
set_state(waiting_for_readable);
|
set_state(waiting_for_readable);
|
||||||
} else if (get_time() >= state_start_usec + k_readable_too_long_duration_usec) {
|
} else if (get_time() >= state_start_usec + k_readable_too_long_duration_usec) {
|
||||||
drain_excess();
|
drain_excess();
|
||||||
|
@ -1479,7 +1479,7 @@ class universal_notifier_named_pipe_t final : public universal_notifier_t {
|
||||||
// change occurred with ours.
|
// change occurred with ours.
|
||||||
if (get_time() >= state_start_usec + k_flash_duration_usec) {
|
if (get_time() >= state_start_usec + k_flash_duration_usec) {
|
||||||
drain_written();
|
drain_written();
|
||||||
if (!select_wrapper_t::poll_fd_readable(pipe_fd.fd())) {
|
if (!fd_readable_set_t::poll_fd_readable(pipe_fd.fd())) {
|
||||||
set_state(waiting_for_readable);
|
set_state(waiting_for_readable);
|
||||||
} else {
|
} else {
|
||||||
set_state(polling_during_readable);
|
set_state(polling_during_readable);
|
||||||
|
|
|
@ -87,7 +87,7 @@ uint64_t fd_monitor_item_t::usec_remaining(const time_point_t &now) const {
|
||||||
return since >= timeout_usec ? 0 : timeout_usec - since;
|
return since >= timeout_usec ? 0 : timeout_usec - since;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool fd_monitor_item_t::service_item(const select_wrapper_t &fds, const time_point_t &now) {
|
bool fd_monitor_item_t::service_item(const fd_readable_set_t &fds, const time_point_t &now) {
|
||||||
bool should_retain = true;
|
bool should_retain = true;
|
||||||
bool readable = fds.test(fd.fd());
|
bool readable = fds.test(fd.fd());
|
||||||
bool timed_out = !readable && usec_remaining(now) == 0;
|
bool timed_out = !readable && usec_remaining(now) == 0;
|
||||||
|
@ -113,7 +113,7 @@ bool fd_monitor_item_t::poke_item(const poke_list_t &pokelist) {
|
||||||
void fd_monitor_t::run_in_background() {
|
void fd_monitor_t::run_in_background() {
|
||||||
ASSERT_IS_BACKGROUND_THREAD();
|
ASSERT_IS_BACKGROUND_THREAD();
|
||||||
poke_list_t pokelist;
|
poke_list_t pokelist;
|
||||||
select_wrapper_t fds;
|
fd_readable_set_t fds;
|
||||||
for (;;) {
|
for (;;) {
|
||||||
// Poke any items that need it.
|
// Poke any items that need it.
|
||||||
if (!pokelist.empty()) {
|
if (!pokelist.empty()) {
|
||||||
|
@ -149,7 +149,7 @@ void fd_monitor_t::run_in_background() {
|
||||||
}
|
}
|
||||||
|
|
||||||
// Call select().
|
// Call select().
|
||||||
int ret = fds.select(timeout_usec);
|
int ret = fds.check_readable(timeout_usec);
|
||||||
if (ret < 0 && errno != EINTR) {
|
if (ret < 0 && errno != EINTR) {
|
||||||
// Surprising error.
|
// Surprising error.
|
||||||
wperror(L"select");
|
wperror(L"select");
|
||||||
|
|
|
@ -33,7 +33,7 @@ struct fd_monitor_item_t {
|
||||||
using callback_t = std::function<void(autoclose_fd_t &fd, item_wake_reason_t reason)>;
|
using callback_t = std::function<void(autoclose_fd_t &fd, item_wake_reason_t reason)>;
|
||||||
|
|
||||||
/// A sentinel value meaning no timeout.
|
/// A sentinel value meaning no timeout.
|
||||||
static constexpr uint64_t kNoTimeout = select_wrapper_t::kNoTimeout;
|
static constexpr uint64_t kNoTimeout = fd_readable_set_t::kNoTimeout;
|
||||||
|
|
||||||
/// The fd to monitor.
|
/// The fd to monitor.
|
||||||
autoclose_fd_t fd{};
|
autoclose_fd_t fd{};
|
||||||
|
@ -71,7 +71,7 @@ struct fd_monitor_item_t {
|
||||||
|
|
||||||
// Invoke this item's callback if its value is set in fd or has timed out.
|
// Invoke this item's callback if its value is set in fd or has timed out.
|
||||||
// \return true to retain the item, false to remove it.
|
// \return true to retain the item, false to remove it.
|
||||||
bool service_item(const select_wrapper_t &fds, const time_point_t &now);
|
bool service_item(const fd_readable_set_t &fds, const time_point_t &now);
|
||||||
|
|
||||||
// Invoke this item's callback with a poke, if its ID is present in the (sorted) pokelist.
|
// Invoke this item's callback with a poke, if its ID is present in the (sorted) pokelist.
|
||||||
// \return true to retain the item, false to remove it.
|
// \return true to retain the item, false to remove it.
|
||||||
|
|
85
src/fds.cpp
85
src/fds.cpp
|
@ -20,9 +20,8 @@
|
||||||
// The first fd in the "high range." fds below this are allowed to be used directly by users in
|
// The first fd in the "high range." fds below this are allowed to be used directly by users in
|
||||||
// redirections, e.g. >&3
|
// redirections, e.g. >&3
|
||||||
const int k_first_high_fd = 10;
|
const int k_first_high_fd = 10;
|
||||||
|
|
||||||
static constexpr uint64_t kUsecPerMsec = 1000;
|
static constexpr uint64_t kUsecPerMsec = 1000;
|
||||||
static constexpr uint64_t kUsecPerSec = 1000 * kUsecPerMsec;
|
static constexpr uint64_t kUsecPerSec [[gnu::unused]] = 1000 * kUsecPerMsec;
|
||||||
|
|
||||||
void autoclose_fd_t::close() {
|
void autoclose_fd_t::close() {
|
||||||
if (fd_ < 0) return;
|
if (fd_ < 0) return;
|
||||||
|
@ -30,23 +29,85 @@ void autoclose_fd_t::close() {
|
||||||
fd_ = -1;
|
fd_ = -1;
|
||||||
}
|
}
|
||||||
|
|
||||||
select_wrapper_t::select_wrapper_t() { clear(); }
|
fd_readable_set_t::fd_readable_set_t() { clear(); }
|
||||||
|
|
||||||
void select_wrapper_t::clear() {
|
#if FISH_READABLE_SET_USE_POLL
|
||||||
|
|
||||||
|
// Convert from a usec to a poll-friendly msec.
|
||||||
|
static int usec_to_poll_msec(uint64_t timeout_usec) {
|
||||||
|
uint64_t timeout_msec = timeout_usec / kUsecPerMsec;
|
||||||
|
// Round to nearest, down for halfway.
|
||||||
|
timeout_msec += ((timeout_usec % kUsecPerMsec) > kUsecPerMsec / 2) ? 1 : 0;
|
||||||
|
if (timeout_usec == fd_readable_set_t::kNoTimeout ||
|
||||||
|
timeout_msec > std::numeric_limits<int>::max()) {
|
||||||
|
// Negative values mean wait forever in poll-speak.
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
return static_cast<int>(timeout_msec);
|
||||||
|
}
|
||||||
|
|
||||||
|
void fd_readable_set_t::clear() { pollfds_.clear(); }
|
||||||
|
|
||||||
|
static inline bool pollfd_less_than(const pollfd &lhs, int rhs) { return lhs.fd < rhs; }
|
||||||
|
|
||||||
|
void fd_readable_set_t::add(int fd) {
|
||||||
|
if (fd >= 0) {
|
||||||
|
auto where = std::lower_bound(pollfds_.begin(), pollfds_.end(), fd, pollfd_less_than);
|
||||||
|
if (where == pollfds_.end() || where->fd != fd) {
|
||||||
|
pollfds_.insert(where, pollfd{fd, POLLIN, 0});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
bool fd_readable_set_t::test(int fd) const {
|
||||||
|
// If a pipe is widowed with no data, Linux sets POLLHUP but not POLLIN, so test for both.
|
||||||
|
auto where = std::lower_bound(pollfds_.begin(), pollfds_.end(), fd, pollfd_less_than);
|
||||||
|
return where != pollfds_.end() && where->fd == fd && (where->revents & (POLLIN | POLLHUP));
|
||||||
|
}
|
||||||
|
|
||||||
|
// static
|
||||||
|
int fd_readable_set_t::do_poll(struct pollfd *fds, size_t count, uint64_t timeout_usec) {
|
||||||
|
assert(count <= std::numeric_limits<nfds_t>::max() && "count too big");
|
||||||
|
return ::poll(fds, static_cast<nfds_t>(count), usec_to_poll_msec(timeout_usec));
|
||||||
|
}
|
||||||
|
|
||||||
|
int fd_readable_set_t::check_readable(uint64_t timeout_usec) {
|
||||||
|
if (pollfds_.empty()) return 0;
|
||||||
|
return do_poll(&pollfds_[0], pollfds_.size(), timeout_usec);
|
||||||
|
}
|
||||||
|
|
||||||
|
// static
|
||||||
|
bool fd_readable_set_t::is_fd_readable(int fd, uint64_t timeout_usec) {
|
||||||
|
if (fd < 0) return false;
|
||||||
|
struct pollfd pfd {
|
||||||
|
fd, POLLIN, 0
|
||||||
|
};
|
||||||
|
int ret = fd_readable_set_t::do_poll(&pfd, 1, timeout_usec);
|
||||||
|
return ret > 0 && (pfd.revents & POLLIN);
|
||||||
|
}
|
||||||
|
|
||||||
|
#else
|
||||||
|
// Implementation based on select().
|
||||||
|
|
||||||
|
void fd_readable_set_t::clear() {
|
||||||
FD_ZERO(&fdset_);
|
FD_ZERO(&fdset_);
|
||||||
nfds_ = 0;
|
nfds_ = 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
void select_wrapper_t::add(int fd) {
|
void fd_readable_set_t::add(int fd) {
|
||||||
|
if (fd >= FD_SETSIZE) {
|
||||||
|
FLOGF(error, "fd %d too large for select()", fd);
|
||||||
|
return;
|
||||||
|
}
|
||||||
if (fd >= 0) {
|
if (fd >= 0) {
|
||||||
FD_SET(fd, &fdset_);
|
FD_SET(fd, &fdset_);
|
||||||
nfds_ = std::max(nfds_, fd + 1);
|
nfds_ = std::max(nfds_, fd + 1);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
bool select_wrapper_t::test(int fd) const { return fd >= 0 && FD_ISSET(fd, &fdset_); }
|
bool fd_readable_set_t::test(int fd) const { return fd >= 0 && FD_ISSET(fd, &fdset_); }
|
||||||
|
|
||||||
int select_wrapper_t::select(uint64_t timeout_usec) {
|
int fd_readable_set_t::check_readable(uint64_t timeout_usec) {
|
||||||
if (timeout_usec == kNoTimeout) {
|
if (timeout_usec == kNoTimeout) {
|
||||||
return ::select(nfds_, &fdset_, nullptr, nullptr, nullptr);
|
return ::select(nfds_, &fdset_, nullptr, nullptr, nullptr);
|
||||||
} else {
|
} else {
|
||||||
|
@ -58,16 +119,18 @@ int select_wrapper_t::select(uint64_t timeout_usec) {
|
||||||
}
|
}
|
||||||
|
|
||||||
// static
|
// static
|
||||||
bool select_wrapper_t::is_fd_readable(int fd, uint64_t timeout_usec) {
|
bool fd_readable_set_t::is_fd_readable(int fd, uint64_t timeout_usec) {
|
||||||
if (fd < 0) return false;
|
if (fd < 0) return false;
|
||||||
select_wrapper_t s;
|
fd_readable_set_t s;
|
||||||
s.add(fd);
|
s.add(fd);
|
||||||
int res = s.select(timeout_usec);
|
int res = s.check_readable(timeout_usec);
|
||||||
return res > 0 && s.test(fd);
|
return res > 0 && s.test(fd);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#endif // not FISH_READABLE_SET_USE_POLL
|
||||||
|
|
||||||
// static
|
// static
|
||||||
bool select_wrapper_t::poll_fd_readable(int fd) { return is_fd_readable(fd, 0); }
|
bool fd_readable_set_t::poll_fd_readable(int fd) { return is_fd_readable(fd, 0); }
|
||||||
|
|
||||||
#ifdef HAVE_EVENTFD
|
#ifdef HAVE_EVENTFD
|
||||||
// Note we do not want to use EFD_SEMAPHORE because we are binary (not counting) semaphore.
|
// Note we do not want to use EFD_SEMAPHORE because we are binary (not counting) semaphore.
|
||||||
|
|
40
src/fds.h
40
src/fds.h
|
@ -5,6 +5,7 @@
|
||||||
|
|
||||||
#include "config.h" // IWYU pragma: keep
|
#include "config.h" // IWYU pragma: keep
|
||||||
|
|
||||||
|
#include <poll.h>
|
||||||
#include <sys/select.h>
|
#include <sys/select.h>
|
||||||
#include <sys/types.h>
|
#include <sys/types.h>
|
||||||
|
|
||||||
|
@ -62,12 +63,23 @@ class autoclose_fd_t : noncopyable_t {
|
||||||
~autoclose_fd_t() { close(); }
|
~autoclose_fd_t() { close(); }
|
||||||
};
|
};
|
||||||
|
|
||||||
/// A modest wrapper around fd_set and select().
|
// Resolve whether to use poll() or select().
|
||||||
/// This allows accumulating a set of fds and then select()ing on them.
|
#ifndef FISH_READABLE_SET_USE_POLL
|
||||||
|
#ifdef __APPLE__
|
||||||
|
// Apple's `man poll`: "The poll() system call currently does not support devices."
|
||||||
|
#define FISH_READABLE_SET_USE_POLL 0
|
||||||
|
#else
|
||||||
|
// Use poll other places so we can support unlimited fds.
|
||||||
|
#define FISH_READABLE_SET_USE_POLL 1
|
||||||
|
#endif
|
||||||
|
#endif
|
||||||
|
|
||||||
|
/// A modest wrapper around select() or poll(), according to FISH_READABLE_SET_USE_POLL.
|
||||||
|
/// This allows accumulating a set of fds and then seeing if they are readable.
|
||||||
/// This only handles readability.
|
/// This only handles readability.
|
||||||
struct select_wrapper_t {
|
struct fd_readable_set_t {
|
||||||
/// Construct an empty set.
|
/// Construct an empty set.
|
||||||
select_wrapper_t();
|
fd_readable_set_t();
|
||||||
|
|
||||||
/// Reset back to an empty set.
|
/// Reset back to an empty set.
|
||||||
void clear();
|
void clear();
|
||||||
|
@ -78,15 +90,15 @@ struct select_wrapper_t {
|
||||||
/// \return true if the given fd is marked as set, in our set. \returns false if negative.
|
/// \return true if the given fd is marked as set, in our set. \returns false if negative.
|
||||||
bool test(int fd) const;
|
bool test(int fd) const;
|
||||||
|
|
||||||
/// Call select(), with this set as 'readfds' and null for the other sets, with a timeout given
|
/// Call select() or poll(), according to FISH_READABLE_SET_USE_POLL. Note this destructively
|
||||||
/// by timeout_usec. Note this destructively modifies the set. \return the result of select().
|
/// modifies the set. \return the result of select() or poll().
|
||||||
int select(uint64_t timeout_usec = select_wrapper_t::kNoTimeout);
|
int check_readable(uint64_t timeout_usec = fd_readable_set_t::kNoTimeout);
|
||||||
|
|
||||||
/// Poll a single fd: select() on it with a given timeout.
|
/// Check if a single fd is readable, with a given timeout.
|
||||||
/// \return true if readable, false if not.
|
/// \return true if readable, false if not.
|
||||||
static bool is_fd_readable(int fd, uint64_t timeout_usec);
|
static bool is_fd_readable(int fd, uint64_t timeout_usec);
|
||||||
|
|
||||||
/// Poll a single fd: select() on it with zero timeout.
|
/// Check if a single fd is readable, without blocking.
|
||||||
/// \return true if readable, false if not.
|
/// \return true if readable, false if not.
|
||||||
static bool poll_fd_readable(int fd);
|
static bool poll_fd_readable(int fd);
|
||||||
|
|
||||||
|
@ -94,9 +106,17 @@ struct select_wrapper_t {
|
||||||
static constexpr uint64_t kNoTimeout = std::numeric_limits<uint64_t>::max();
|
static constexpr uint64_t kNoTimeout = std::numeric_limits<uint64_t>::max();
|
||||||
|
|
||||||
private:
|
private:
|
||||||
/// The underlying fdset and nfds value to pass to select().
|
#if FISH_READABLE_SET_USE_POLL
|
||||||
|
// Our list of FDs, sorted by fd.
|
||||||
|
std::vector<struct pollfd> pollfds_{};
|
||||||
|
|
||||||
|
// Helper function.
|
||||||
|
static int do_poll(struct pollfd *fds, size_t count, uint64_t timeout_usec);
|
||||||
|
#else
|
||||||
|
// The underlying fdset and nfds value to pass to select().
|
||||||
fd_set fdset_;
|
fd_set fdset_;
|
||||||
int nfds_{0};
|
int nfds_{0};
|
||||||
|
#endif
|
||||||
};
|
};
|
||||||
|
|
||||||
/// Helper type returned from making autoclose pipes.
|
/// Helper type returned from making autoclose pipes.
|
||||||
|
|
|
@ -4131,7 +4131,7 @@ bool poll_notifier(const std::unique_ptr<universal_notifier_t> ¬e) {
|
||||||
|
|
||||||
bool result = false;
|
bool result = false;
|
||||||
int fd = note->notification_fd();
|
int fd = note->notification_fd();
|
||||||
if (fd >= 0 && select_wrapper_t::poll_fd_readable(fd)) {
|
if (fd >= 0 && fd_readable_set_t::poll_fd_readable(fd)) {
|
||||||
result = note->notification_fd_became_readable(fd);
|
result = note->notification_fd_became_readable(fd);
|
||||||
}
|
}
|
||||||
return result;
|
return result;
|
||||||
|
|
|
@ -60,7 +60,7 @@ using readb_result_t = int;
|
||||||
static readb_result_t readb(int in_fd) {
|
static readb_result_t readb(int in_fd) {
|
||||||
assert(in_fd >= 0 && "Invalid in fd");
|
assert(in_fd >= 0 && "Invalid in fd");
|
||||||
universal_notifier_t& notifier = universal_notifier_t::default_notifier();
|
universal_notifier_t& notifier = universal_notifier_t::default_notifier();
|
||||||
select_wrapper_t fdset;
|
fd_readable_set_t fdset;
|
||||||
for (;;) {
|
for (;;) {
|
||||||
fdset.clear();
|
fdset.clear();
|
||||||
fdset.add(in_fd);
|
fdset.add(in_fd);
|
||||||
|
@ -75,13 +75,13 @@ static readb_result_t readb(int in_fd) {
|
||||||
|
|
||||||
// Get its suggested delay (possibly none).
|
// Get its suggested delay (possibly none).
|
||||||
// Note a 0 here means do not poll.
|
// Note a 0 here means do not poll.
|
||||||
uint64_t timeout = select_wrapper_t::kNoTimeout;
|
uint64_t timeout = fd_readable_set_t::kNoTimeout;
|
||||||
if (uint64_t usecs_delay = notifier.usec_delay_between_polls()) {
|
if (uint64_t usecs_delay = notifier.usec_delay_between_polls()) {
|
||||||
timeout = usecs_delay;
|
timeout = usecs_delay;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Here's where we call select().
|
// Here's where we call select().
|
||||||
int select_res = fdset.select(timeout);
|
int select_res = fdset.check_readable(timeout);
|
||||||
if (select_res < 0) {
|
if (select_res < 0) {
|
||||||
if (errno == EINTR || errno == EAGAIN) {
|
if (errno == EINTR || errno == EAGAIN) {
|
||||||
// A signal.
|
// A signal.
|
||||||
|
@ -225,7 +225,7 @@ maybe_t<char_event_t> input_event_queue_t::readch_timed() {
|
||||||
}
|
}
|
||||||
const uint64_t usec_per_msec = 1000;
|
const uint64_t usec_per_msec = 1000;
|
||||||
uint64_t timeout_usec = static_cast<uint64_t>(wait_on_escape_ms) * usec_per_msec;
|
uint64_t timeout_usec = static_cast<uint64_t>(wait_on_escape_ms) * usec_per_msec;
|
||||||
if (select_wrapper_t::is_fd_readable(in_, timeout_usec)) {
|
if (fd_readable_set_t::is_fd_readable(in_, timeout_usec)) {
|
||||||
return readch();
|
return readch();
|
||||||
}
|
}
|
||||||
return none();
|
return none();
|
||||||
|
|
|
@ -247,7 +247,7 @@ void iothread_perform_impl(void_function_t &&func, bool cant_wait) {
|
||||||
int iothread_port() { return get_notify_signaller().read_fd(); }
|
int iothread_port() { return get_notify_signaller().read_fd(); }
|
||||||
|
|
||||||
void iothread_service_main_with_timeout(uint64_t timeout_usec) {
|
void iothread_service_main_with_timeout(uint64_t timeout_usec) {
|
||||||
if (select_wrapper_t::is_fd_readable(iothread_port(), timeout_usec)) {
|
if (fd_readable_set_t::is_fd_readable(iothread_port(), timeout_usec)) {
|
||||||
iothread_service_main();
|
iothread_service_main();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -2935,7 +2935,7 @@ maybe_t<char_event_t> reader_data_t::read_normal_chars(readline_loop_state_t &rl
|
||||||
while (accumulated_chars.size() < limit) {
|
while (accumulated_chars.size() < limit) {
|
||||||
bool allow_commands = (accumulated_chars.empty());
|
bool allow_commands = (accumulated_chars.empty());
|
||||||
auto evt = inputter.read_char(allow_commands ? normal_handler : empty_handler);
|
auto evt = inputter.read_char(allow_commands ? normal_handler : empty_handler);
|
||||||
if (!event_is_normal_char(evt) || !select_wrapper_t::poll_fd_readable(conf.in)) {
|
if (!event_is_normal_char(evt) || !fd_readable_set_t::poll_fd_readable(conf.in)) {
|
||||||
event_needing_handling = std::move(evt);
|
event_needing_handling = std::move(evt);
|
||||||
break;
|
break;
|
||||||
} else if (evt.input_style == char_input_style_t::notfirst && accumulated_chars.empty() &&
|
} else if (evt.input_style == char_input_style_t::notfirst && accumulated_chars.empty() &&
|
||||||
|
|
|
@ -89,8 +89,8 @@ void binary_semaphore_t::wait() {
|
||||||
#ifdef FISH_TSAN_WORKAROUNDS
|
#ifdef FISH_TSAN_WORKAROUNDS
|
||||||
// Under tsan our notifying pipe is non-blocking, so we would busy-loop on the read()
|
// Under tsan our notifying pipe is non-blocking, so we would busy-loop on the read()
|
||||||
// call until data is available (that is, fish would use 100% cpu while waiting for
|
// call until data is available (that is, fish would use 100% cpu while waiting for
|
||||||
// processes). The select prevents that.
|
// processes). This call prevents that.
|
||||||
(void)select_wrapper_t::is_fd_readable(fd, select_wrapper_t::kNoTimeout);
|
(void)fd_readable_set_t::is_fd_readable(fd, fd_readable_set_t::kNoTimeout);
|
||||||
#endif
|
#endif
|
||||||
uint8_t ignored;
|
uint8_t ignored;
|
||||||
auto amt = read(fd, &ignored, sizeof ignored);
|
auto amt = read(fd, &ignored, sizeof ignored);
|
||||||
|
|
Loading…
Reference in a new issue