summaryrefslogtreecommitdiff
path: root/safe_poll.c
diff options
context:
space:
mode:
authorMichele Bini <rev@alienbuntu.local>2019-04-18 22:12:22 (GMT)
committerMichele Bini <rev@alienbuntu.local>2019-04-18 22:13:11 (GMT)
commit3567ae383ea8d19f054ba138c8250446cc4a8bb1 (patch)
treed7d23be597c2ffb7176fbaa511050a067947a62f /safe_poll.c
parentda7ba579a7d9a95457f87bc5389a53b12d6fc9cb (diff)
.
Diffstat (limited to 'safe_poll.c')
-rw-r--r--safe_poll.c34
1 files changed, 34 insertions, 0 deletions
diff --git a/safe_poll.c b/safe_poll.c
new file mode 100644
index 0000000..b492a81
--- /dev/null
+++ b/safe_poll.c
@@ -0,0 +1,34 @@
+/* vi: set sw=4 ts=4: */
+/*
+ * Utility routines.
+ *
+ * Copyright (C) 2007 by Denys Vlasenko <vda.linux@googlemail.com>
+ *
+ * Licensed under GPLv2, see file LICENSE in this source tree.
+ */
+
+#include "libbb.h"
+
+/* Wrapper which restarts poll on EINTR or ENOMEM.
+ * On other errors does perror("poll") and returns.
+ * Warning! May take longer than timeout_ms to return! */
+int FAST_FUNC safe_poll(struct pollfd *ufds, nfds_t nfds, int timeout)
+{
+ while (1) {
+ int n = poll(ufds, nfds, timeout);
+ if (n >= 0)
+ return n;
+ /* Make sure we inch towards completion */
+ if (timeout > 0)
+ timeout--;
+ /* E.g. strace causes poll to return this */
+ if (errno == EINTR)
+ continue;
+ /* Kernel is very low on memory. Retry. */
+ /* I doubt many callers would handle this correctly! */
+ if (errno == ENOMEM)
+ continue;
+ bb_perror_msg("poll");
+ return n;
+ }
+}