libexplain  1.4.D001
libexplain/nanosleep_or_die.c
Go to the documentation of this file.
00001 /*
00002  * libexplain - Explain errno values returned by libc functions
00003  * Copyright (C) 2013 Peter Miller
00004  *
00005  * This program is free software; you can redistribute it and/or modify it
00006  * under the terms of the GNU Lesser General Public License as published by
00007  * the Free Software Foundation; either version 3 of the License, or (at
00008  * your option) any later version.
00009  *
00010  * This program is distributed in the hope that it will be useful, but
00011  * WITHOUT ANY WARRANTY; without even the implied warranty of
00012  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU Lesser
00013  * General Public License for more details.
00014  *
00015  * You should have received a copy of the GNU Lesser General Public License
00016  * along with this program. If not, see <http://www.gnu.org/licenses/>.
00017  */
00018 
00019 #include <libexplain/ac/errno.h>
00020 #include <libexplain/ac/time.h>
00021 
00022 #include <libexplain/nanosleep.h>
00023 #include <libexplain/output.h>
00024 
00025 
00026 void
00027 explain_nanosleep_or_die(const struct timespec *req, struct timespec *rem)
00028 {
00029     if (explain_nanosleep_on_error(req, rem) < 0)
00030     {
00031         explain_output_exit_failure();
00032     }
00033 }
00034 
00035 
00036 int
00037 explain_nanosleep_on_error(const struct timespec *req, struct timespec *rem)
00038 {
00039     int             hold_errno;
00040     int             result;
00041 
00042     hold_errno = errno;
00043     errno = 0;
00044 #ifdef HAVE_NANOSLEEP
00045     result = nanosleep(req, rem);
00046 #else
00047     errno = ENOSYS;
00048     result = -1;
00049 #endif
00050     if (result < 0)
00051     {
00052         /*
00053          * Linux-2.6.8.1's nanosleep returns -1, but doesn't set errno
00054          * when resumed after being suspended.  Earlier versions would
00055          * set errno to EINTR. nanosleep from linux-2.6.10, as well as
00056          * implementations by (all?) other vendors, doesn't return -1 in
00057          * that case; either it continues sleeping (if time remains. or
00058          * it returns zero (if the wake-up time has passed).
00059          */
00060         hold_errno = errno ? errno : EINTR;
00061         explain_output_error
00062         (
00063             "%s",
00064             explain_errno_nanosleep(hold_errno, req, rem)
00065         );
00066     }
00067     errno = hold_errno;
00068     return result;
00069 }
00070 
00071 
00072 /* vim: set ts=8 sw=4 et : */