|  | /* 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; | 
|  | } |