183d2307dSDag-Erling Smørgrav /* 2d95e11bfSDag-Erling Smørgrav * Copyright (c) 1999-2003 Damien Miller. All rights reserved. 383d2307dSDag-Erling Smørgrav * 483d2307dSDag-Erling Smørgrav * Redistribution and use in source and binary forms, with or without 583d2307dSDag-Erling Smørgrav * modification, are permitted provided that the following conditions 683d2307dSDag-Erling Smørgrav * are met: 783d2307dSDag-Erling Smørgrav * 1. Redistributions of source code must retain the above copyright 883d2307dSDag-Erling Smørgrav * notice, this list of conditions and the following disclaimer. 983d2307dSDag-Erling Smørgrav * 2. Redistributions in binary form must reproduce the above copyright 1083d2307dSDag-Erling Smørgrav * notice, this list of conditions and the following disclaimer in the 1183d2307dSDag-Erling Smørgrav * documentation and/or other materials provided with the distribution. 1283d2307dSDag-Erling Smørgrav * 1383d2307dSDag-Erling Smørgrav * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR 1483d2307dSDag-Erling Smørgrav * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES 1583d2307dSDag-Erling Smørgrav * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. 1683d2307dSDag-Erling Smørgrav * IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT, 1783d2307dSDag-Erling Smørgrav * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT 1883d2307dSDag-Erling Smørgrav * NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, 1983d2307dSDag-Erling Smørgrav * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY 2083d2307dSDag-Erling Smørgrav * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT 2183d2307dSDag-Erling Smørgrav * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF 2283d2307dSDag-Erling Smørgrav * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. 2383d2307dSDag-Erling Smørgrav */ 2483d2307dSDag-Erling Smørgrav 2583d2307dSDag-Erling Smørgrav #include "includes.h" 26d0c8c0bcSDag-Erling Smørgrav #include "xmalloc.h" 2783d2307dSDag-Erling Smørgrav 28d95e11bfSDag-Erling Smørgrav RCSID("$Id: bsd-misc.c,v 1.19 2003/08/25 01:16:21 mouring Exp $"); 2983d2307dSDag-Erling Smørgrav 30d0c8c0bcSDag-Erling Smørgrav /* 31d0c8c0bcSDag-Erling Smørgrav * NB. duplicate __progname in case it is an alias for argv[0] 32d0c8c0bcSDag-Erling Smørgrav * Otherwise it may get clobbered by setproctitle() 33d0c8c0bcSDag-Erling Smørgrav */ 34d95e11bfSDag-Erling Smørgrav char *ssh_get_progname(char *argv0) 3583d2307dSDag-Erling Smørgrav { 3683d2307dSDag-Erling Smørgrav #ifdef HAVE___PROGNAME 3783d2307dSDag-Erling Smørgrav extern char *__progname; 3883d2307dSDag-Erling Smørgrav 39d0c8c0bcSDag-Erling Smørgrav return xstrdup(__progname); 4083d2307dSDag-Erling Smørgrav #else 4183d2307dSDag-Erling Smørgrav char *p; 4283d2307dSDag-Erling Smørgrav 4383d2307dSDag-Erling Smørgrav if (argv0 == NULL) 44d95e11bfSDag-Erling Smørgrav return ("unknown"); /* XXX */ 4583d2307dSDag-Erling Smørgrav p = strrchr(argv0, '/'); 4683d2307dSDag-Erling Smørgrav if (p == NULL) 4783d2307dSDag-Erling Smørgrav p = argv0; 4883d2307dSDag-Erling Smørgrav else 4983d2307dSDag-Erling Smørgrav p++; 50d0c8c0bcSDag-Erling Smørgrav 51d95e11bfSDag-Erling Smørgrav return (xstrdup(p)); 5283d2307dSDag-Erling Smørgrav #endif 5383d2307dSDag-Erling Smørgrav } 5483d2307dSDag-Erling Smørgrav 5583d2307dSDag-Erling Smørgrav #ifndef HAVE_SETLOGIN 5683d2307dSDag-Erling Smørgrav int setlogin(const char *name) 5783d2307dSDag-Erling Smørgrav { 5883d2307dSDag-Erling Smørgrav return (0); 5983d2307dSDag-Erling Smørgrav } 6083d2307dSDag-Erling Smørgrav #endif /* !HAVE_SETLOGIN */ 6183d2307dSDag-Erling Smørgrav 6283d2307dSDag-Erling Smørgrav #ifndef HAVE_INNETGR 6383d2307dSDag-Erling Smørgrav int innetgr(const char *netgroup, const char *host, 6483d2307dSDag-Erling Smørgrav const char *user, const char *domain) 6583d2307dSDag-Erling Smørgrav { 6683d2307dSDag-Erling Smørgrav return (0); 6783d2307dSDag-Erling Smørgrav } 6883d2307dSDag-Erling Smørgrav #endif /* HAVE_INNETGR */ 6983d2307dSDag-Erling Smørgrav 7083d2307dSDag-Erling Smørgrav #if !defined(HAVE_SETEUID) && defined(HAVE_SETREUID) 7183d2307dSDag-Erling Smørgrav int seteuid(uid_t euid) 7283d2307dSDag-Erling Smørgrav { 7383d2307dSDag-Erling Smørgrav return (setreuid(-1, euid)); 7483d2307dSDag-Erling Smørgrav } 7583d2307dSDag-Erling Smørgrav #endif /* !defined(HAVE_SETEUID) && defined(HAVE_SETREUID) */ 7683d2307dSDag-Erling Smørgrav 7783d2307dSDag-Erling Smørgrav #if !defined(HAVE_SETEGID) && defined(HAVE_SETRESGID) 7883d2307dSDag-Erling Smørgrav int setegid(uid_t egid) 7983d2307dSDag-Erling Smørgrav { 8083d2307dSDag-Erling Smørgrav return(setresgid(-1, egid, -1)); 8183d2307dSDag-Erling Smørgrav } 8283d2307dSDag-Erling Smørgrav #endif /* !defined(HAVE_SETEGID) && defined(HAVE_SETRESGID) */ 8383d2307dSDag-Erling Smørgrav 8483d2307dSDag-Erling Smørgrav #if !defined(HAVE_STRERROR) && defined(HAVE_SYS_ERRLIST) && defined(HAVE_SYS_NERR) 8583d2307dSDag-Erling Smørgrav const char *strerror(int e) 8683d2307dSDag-Erling Smørgrav { 8783d2307dSDag-Erling Smørgrav extern int sys_nerr; 8883d2307dSDag-Erling Smørgrav extern char *sys_errlist[]; 8983d2307dSDag-Erling Smørgrav 9083d2307dSDag-Erling Smørgrav if ((e >= 0) && (e < sys_nerr)) 9183d2307dSDag-Erling Smørgrav return (sys_errlist[e]); 92d95e11bfSDag-Erling Smørgrav 9383d2307dSDag-Erling Smørgrav return ("unlisted error"); 9483d2307dSDag-Erling Smørgrav } 9583d2307dSDag-Erling Smørgrav #endif 9683d2307dSDag-Erling Smørgrav 9783d2307dSDag-Erling Smørgrav #ifndef HAVE_UTIMES 9883d2307dSDag-Erling Smørgrav int utimes(char *filename, struct timeval *tvp) 9983d2307dSDag-Erling Smørgrav { 10083d2307dSDag-Erling Smørgrav struct utimbuf ub; 10183d2307dSDag-Erling Smørgrav 1024b17dab0SDag-Erling Smørgrav ub.actime = tvp[0].tv_sec; 1034b17dab0SDag-Erling Smørgrav ub.modtime = tvp[1].tv_sec; 10483d2307dSDag-Erling Smørgrav 10583d2307dSDag-Erling Smørgrav return (utime(filename, &ub)); 10683d2307dSDag-Erling Smørgrav } 10783d2307dSDag-Erling Smørgrav #endif 10883d2307dSDag-Erling Smørgrav 10983d2307dSDag-Erling Smørgrav #ifndef HAVE_TRUNCATE 11083d2307dSDag-Erling Smørgrav int truncate(const char *path, off_t length) 11183d2307dSDag-Erling Smørgrav { 11283d2307dSDag-Erling Smørgrav int fd, ret, saverrno; 11383d2307dSDag-Erling Smørgrav 11483d2307dSDag-Erling Smørgrav fd = open(path, O_WRONLY); 11583d2307dSDag-Erling Smørgrav if (fd < 0) 116d95e11bfSDag-Erling Smørgrav return (-1); 11783d2307dSDag-Erling Smørgrav 11883d2307dSDag-Erling Smørgrav ret = ftruncate(fd, length); 11983d2307dSDag-Erling Smørgrav saverrno = errno; 120d95e11bfSDag-Erling Smørgrav close(fd); 12183d2307dSDag-Erling Smørgrav if (ret == -1) 12283d2307dSDag-Erling Smørgrav errno = saverrno; 123d95e11bfSDag-Erling Smørgrav 12483d2307dSDag-Erling Smørgrav return(ret); 12583d2307dSDag-Erling Smørgrav } 12683d2307dSDag-Erling Smørgrav #endif /* HAVE_TRUNCATE */ 12783d2307dSDag-Erling Smørgrav 12883d2307dSDag-Erling Smørgrav #if !defined(HAVE_SETGROUPS) && defined(SETGROUPS_NOOP) 12983d2307dSDag-Erling Smørgrav /* 13083d2307dSDag-Erling Smørgrav * Cygwin setgroups should be a noop. 13183d2307dSDag-Erling Smørgrav */ 13283d2307dSDag-Erling Smørgrav int 13383d2307dSDag-Erling Smørgrav setgroups(size_t size, const gid_t *list) 13483d2307dSDag-Erling Smørgrav { 135d95e11bfSDag-Erling Smørgrav return (0); 13683d2307dSDag-Erling Smørgrav } 13783d2307dSDag-Erling Smørgrav #endif 13883d2307dSDag-Erling Smørgrav 139d0c8c0bcSDag-Erling Smørgrav #if !defined(HAVE_NANOSLEEP) && !defined(HAVE_NSLEEP) 140d0c8c0bcSDag-Erling Smørgrav int nanosleep(const struct timespec *req, struct timespec *rem) 141d0c8c0bcSDag-Erling Smørgrav { 142d0c8c0bcSDag-Erling Smørgrav int rc, saverrno; 143d0c8c0bcSDag-Erling Smørgrav extern int errno; 144d0c8c0bcSDag-Erling Smørgrav struct timeval tstart, tstop, tremain, time2wait; 145d0c8c0bcSDag-Erling Smørgrav 146d0c8c0bcSDag-Erling Smørgrav TIMESPEC_TO_TIMEVAL(&time2wait, req) 147d0c8c0bcSDag-Erling Smørgrav (void) gettimeofday(&tstart, NULL); 148d0c8c0bcSDag-Erling Smørgrav rc = select(0, NULL, NULL, NULL, &time2wait); 149d0c8c0bcSDag-Erling Smørgrav if (rc == -1) { 150d0c8c0bcSDag-Erling Smørgrav saverrno = errno; 151d0c8c0bcSDag-Erling Smørgrav (void) gettimeofday (&tstop, NULL); 152d0c8c0bcSDag-Erling Smørgrav errno = saverrno; 153d0c8c0bcSDag-Erling Smørgrav tremain.tv_sec = time2wait.tv_sec - 154d0c8c0bcSDag-Erling Smørgrav (tstop.tv_sec - tstart.tv_sec); 155d0c8c0bcSDag-Erling Smørgrav tremain.tv_usec = time2wait.tv_usec - 156d0c8c0bcSDag-Erling Smørgrav (tstop.tv_usec - tstart.tv_usec); 157d0c8c0bcSDag-Erling Smørgrav tremain.tv_sec += tremain.tv_usec / 1000000L; 158d0c8c0bcSDag-Erling Smørgrav tremain.tv_usec %= 1000000L; 159d0c8c0bcSDag-Erling Smørgrav } else { 160d0c8c0bcSDag-Erling Smørgrav tremain.tv_sec = 0; 161d0c8c0bcSDag-Erling Smørgrav tremain.tv_usec = 0; 162d0c8c0bcSDag-Erling Smørgrav } 163d0c8c0bcSDag-Erling Smørgrav TIMEVAL_TO_TIMESPEC(&tremain, rem) 164d0c8c0bcSDag-Erling Smørgrav 165d0c8c0bcSDag-Erling Smørgrav return(rc); 166d0c8c0bcSDag-Erling Smørgrav } 167d0c8c0bcSDag-Erling Smørgrav 168d0c8c0bcSDag-Erling Smørgrav #endif 169d0c8c0bcSDag-Erling Smørgrav 170d95e11bfSDag-Erling Smørgrav #ifndef HAVE_TCGETPGRP 171d95e11bfSDag-Erling Smørgrav pid_t 172d95e11bfSDag-Erling Smørgrav tcgetpgrp(int fd) 173d95e11bfSDag-Erling Smørgrav { 174d95e11bfSDag-Erling Smørgrav int ctty_pgrp; 175d95e11bfSDag-Erling Smørgrav 176d95e11bfSDag-Erling Smørgrav if (ioctl(fd, TIOCGPGRP, &ctty_pgrp) == -1) 177d95e11bfSDag-Erling Smørgrav return(-1); 178d95e11bfSDag-Erling Smørgrav else 179d95e11bfSDag-Erling Smørgrav return(ctty_pgrp); 180d95e11bfSDag-Erling Smørgrav } 181d95e11bfSDag-Erling Smørgrav #endif /* HAVE_TCGETPGRP */ 182d95e11bfSDag-Erling Smørgrav 183d95e11bfSDag-Erling Smørgrav #ifndef HAVE_TCSENDBREAK 184d95e11bfSDag-Erling Smørgrav int 185d95e11bfSDag-Erling Smørgrav tcsendbreak(int fd, int duration) 186d95e11bfSDag-Erling Smørgrav { 187d95e11bfSDag-Erling Smørgrav # if defined(TIOCSBRK) && defined(TIOCCBRK) 188d95e11bfSDag-Erling Smørgrav struct timeval sleepytime; 189d95e11bfSDag-Erling Smørgrav 190d95e11bfSDag-Erling Smørgrav sleepytime.tv_sec = 0; 191d95e11bfSDag-Erling Smørgrav sleepytime.tv_usec = 400000; 192d95e11bfSDag-Erling Smørgrav if (ioctl(fd, TIOCSBRK, 0) == -1) 193d95e11bfSDag-Erling Smørgrav return (-1); 194d95e11bfSDag-Erling Smørgrav (void)select(0, 0, 0, 0, &sleepytime); 195d95e11bfSDag-Erling Smørgrav if (ioctl(fd, TIOCCBRK, 0) == -1) 196d95e11bfSDag-Erling Smørgrav return (-1); 197d95e11bfSDag-Erling Smørgrav return (0); 198d95e11bfSDag-Erling Smørgrav # else 199d95e11bfSDag-Erling Smørgrav return -1; 200d95e11bfSDag-Erling Smørgrav # endif 201d95e11bfSDag-Erling Smørgrav } 202d95e11bfSDag-Erling Smørgrav #endif /* HAVE_TCSENDBREAK */ 203d95e11bfSDag-Erling Smørgrav 204d95e11bfSDag-Erling Smørgrav mysig_t 205d95e11bfSDag-Erling Smørgrav mysignal(int sig, mysig_t act) 206d95e11bfSDag-Erling Smørgrav { 207d95e11bfSDag-Erling Smørgrav #ifdef HAVE_SIGACTION 208d95e11bfSDag-Erling Smørgrav struct sigaction sa, osa; 209d95e11bfSDag-Erling Smørgrav 210d95e11bfSDag-Erling Smørgrav if (sigaction(sig, NULL, &osa) == -1) 211d95e11bfSDag-Erling Smørgrav return (mysig_t) -1; 212d95e11bfSDag-Erling Smørgrav if (osa.sa_handler != act) { 213d95e11bfSDag-Erling Smørgrav memset(&sa, 0, sizeof(sa)); 214d95e11bfSDag-Erling Smørgrav sigemptyset(&sa.sa_mask); 215d95e11bfSDag-Erling Smørgrav sa.sa_flags = 0; 216d95e11bfSDag-Erling Smørgrav #ifdef SA_INTERRUPT 217d95e11bfSDag-Erling Smørgrav if (sig == SIGALRM) 218d95e11bfSDag-Erling Smørgrav sa.sa_flags |= SA_INTERRUPT; 219d95e11bfSDag-Erling Smørgrav #endif 220d95e11bfSDag-Erling Smørgrav sa.sa_handler = act; 221d95e11bfSDag-Erling Smørgrav if (sigaction(sig, &sa, NULL) == -1) 222d95e11bfSDag-Erling Smørgrav return (mysig_t) -1; 223d95e11bfSDag-Erling Smørgrav } 224d95e11bfSDag-Erling Smørgrav return (osa.sa_handler); 225d95e11bfSDag-Erling Smørgrav #else 226d95e11bfSDag-Erling Smørgrav return (signal(sig, act)); 227d95e11bfSDag-Erling Smørgrav #endif 228d95e11bfSDag-Erling Smørgrav } 229