| /* Timeout API for single-threaded programs that use blocking |
| * syscalls (read/write/send/recv/connect/accept). |
| * |
| * Copyright (C) 2017 Red Hat, Inc. |
| * |
| * Author: Stefan Hajnoczi <stefanha@redhat.com> |
| * |
| * This program is free software; you can redistribute it and/or |
| * modify it under the terms of the GNU General Public License |
| * as published by the Free Software Foundation; version 2 |
| * of the License. |
| */ |
| |
| /* Use the following pattern: |
| * |
| * timeout_begin(TIMEOUT); |
| * do { |
| * ret = accept(...); |
| * timeout_check("accept"); |
| * } while (ret < 0 && ret == EINTR); |
| * timeout_end(); |
| */ |
| |
| #include <stdlib.h> |
| #include <stdbool.h> |
| #include <unistd.h> |
| #include <stdio.h> |
| #include "timeout.h" |
| |
| static volatile bool timeout; |
| |
| /* SIGALRM handler function. Do not use sleep(2), alarm(2), or |
| * setitimer(2) while using this API - they may interfere with each |
| * other. |
| */ |
| void sigalrm(int signo) |
| { |
| timeout = true; |
| } |
| |
| /* Start a timeout. Call timeout_check() to verify that the timeout hasn't |
| * expired. timeout_end() must be called to stop the timeout. Timeouts cannot |
| * be nested. |
| */ |
| void timeout_begin(unsigned int seconds) |
| { |
| alarm(seconds); |
| } |
| |
| /* Exit with an error message if the timeout has expired */ |
| void timeout_check(const char *operation) |
| { |
| if (timeout) { |
| fprintf(stderr, "%s timed out\n", operation); |
| exit(EXIT_FAILURE); |
| } |
| } |
| |
| /* Stop a timeout */ |
| void timeout_end(void) |
| { |
| alarm(0); |
| timeout = false; |
| } |