1a5f0fb15SPaul Saab /* 2*c77c4889SXin LI * Copyright (C) 1984-2024 Mark Nudelman 3a5f0fb15SPaul Saab * 4a5f0fb15SPaul Saab * You may distribute under the terms of either the GNU General Public 5a5f0fb15SPaul Saab * License or the Less License, as specified in the README file. 6a5f0fb15SPaul Saab * 796e55cc7SXin LI * For more information, see the README file. 8a5f0fb15SPaul Saab */ 9a5f0fb15SPaul Saab 10a5f0fb15SPaul Saab 11a5f0fb15SPaul Saab /* 12a5f0fb15SPaul Saab * Operating system dependent routines. 13a5f0fb15SPaul Saab * 14a5f0fb15SPaul Saab * Most of the stuff in here is based on Unix, but an attempt 15a5f0fb15SPaul Saab * has been made to make things work on other operating systems. 16a5f0fb15SPaul Saab * This will sometimes result in a loss of functionality, unless 17a5f0fb15SPaul Saab * someone rewrites code specifically for the new operating system. 18a5f0fb15SPaul Saab * 19a5f0fb15SPaul Saab * The makefile provides defines to decide whether various 20a5f0fb15SPaul Saab * Unix features are present. 21a5f0fb15SPaul Saab */ 22a5f0fb15SPaul Saab 23a5f0fb15SPaul Saab #include "less.h" 24a5f0fb15SPaul Saab #include <signal.h> 25a5f0fb15SPaul Saab #include <setjmp.h> 262235c7feSXin LI #if MSDOS_COMPILER==WIN32C 272235c7feSXin LI #include <windows.h> 282235c7feSXin LI #endif 29a5f0fb15SPaul Saab #if HAVE_TIME_H 30a5f0fb15SPaul Saab #include <time.h> 31a5f0fb15SPaul Saab #endif 32a5f0fb15SPaul Saab #if HAVE_ERRNO_H 33a5f0fb15SPaul Saab #include <errno.h> 34a5f0fb15SPaul Saab #endif 35a5f0fb15SPaul Saab #if HAVE_VALUES_H 36a5f0fb15SPaul Saab #include <values.h> 37a5f0fb15SPaul Saab #endif 38a5f0fb15SPaul Saab 39d713e089SXin LI #if defined(__APPLE__) 40d713e089SXin LI #include <sys/utsname.h> 41d713e089SXin LI #endif 42d713e089SXin LI 43d713e089SXin LI #if HAVE_POLL && !MSDOS_COMPILER 442235c7feSXin LI #define USE_POLL 1 45*c77c4889SXin LI static lbool use_poll = TRUE; 462235c7feSXin LI #else 472235c7feSXin LI #define USE_POLL 0 482235c7feSXin LI #endif 492235c7feSXin LI #if USE_POLL 502235c7feSXin LI #include <poll.h> 51*c77c4889SXin LI static lbool any_data = FALSE; 522235c7feSXin LI #endif 532235c7feSXin LI 54a5f0fb15SPaul Saab /* 55a5f0fb15SPaul Saab * BSD setjmp() saves (and longjmp() restores) the signal mask. 56a5f0fb15SPaul Saab * This costs a system call or two per setjmp(), so if possible we clear the 57a5f0fb15SPaul Saab * signal mask with sigsetmask(), and use _setjmp()/_longjmp() instead. 58a5f0fb15SPaul Saab * On other systems, setjmp() doesn't affect the signal mask and so 59a5f0fb15SPaul Saab * _setjmp() does not exist; we just use setjmp(). 60a5f0fb15SPaul Saab */ 61a5f0fb15SPaul Saab #if HAVE__SETJMP && HAVE_SIGSETMASK 62a5f0fb15SPaul Saab #define SET_JUMP _setjmp 63a5f0fb15SPaul Saab #define LONG_JUMP _longjmp 64a5f0fb15SPaul Saab #else 65a5f0fb15SPaul Saab #define SET_JUMP setjmp 66a5f0fb15SPaul Saab #define LONG_JUMP longjmp 67a5f0fb15SPaul Saab #endif 68a5f0fb15SPaul Saab 69a5f0fb15SPaul Saab public int reading; 70*c77c4889SXin LI public lbool waiting_for_data; 7195270f73SXin LI public int consecutive_nulls = 0; 72a5f0fb15SPaul Saab 73d713e089SXin LI /* Milliseconds to wait for data before displaying "waiting for data" message. */ 74d713e089SXin LI static int waiting_for_data_delay = 4000; 75a5f0fb15SPaul Saab static jmp_buf read_label; 76a5f0fb15SPaul Saab 77a5f0fb15SPaul Saab extern int sigs; 782235c7feSXin LI extern int ignore_eoi; 7995270f73SXin LI extern int exit_F_on_close; 80d713e089SXin LI extern int follow_mode; 81d713e089SXin LI extern int scanning_eof; 82d713e089SXin LI extern char intr_char; 83*c77c4889SXin LI extern int is_tty; 842235c7feSXin LI #if !MSDOS_COMPILER 852235c7feSXin LI extern int tty; 862235c7feSXin LI #endif 87d713e089SXin LI 88d713e089SXin LI public void init_poll(void) 89d713e089SXin LI { 90*c77c4889SXin LI constant char *delay = lgetenv("LESS_DATA_DELAY"); 91d713e089SXin LI int idelay = (delay == NULL) ? 0 : atoi(delay); 92d713e089SXin LI if (idelay > 0) 93d713e089SXin LI waiting_for_data_delay = idelay; 94d713e089SXin LI #if USE_POLL 95d713e089SXin LI #if defined(__APPLE__) 96d713e089SXin LI /* In old versions of MacOS, poll() does not work with /dev/tty. */ 97d713e089SXin LI struct utsname uts; 98d713e089SXin LI if (uname(&uts) < 0 || lstrtoi(uts.release, NULL, 10) < 20) 99d713e089SXin LI use_poll = FALSE; 100d713e089SXin LI #endif 101d713e089SXin LI #endif 102d713e089SXin LI } 1032235c7feSXin LI 1042235c7feSXin LI #if USE_POLL 1052235c7feSXin LI /* 106d713e089SXin LI * Check whether data is available, either from a file/pipe or from the tty. 107d713e089SXin LI * Return READ_AGAIN if no data currently available, but caller should retry later. 108d713e089SXin LI * Return READ_INTR to abort F command (forw_loop). 109d713e089SXin LI * Return 0 if safe to read from fd. 1102235c7feSXin LI */ 111d713e089SXin LI static int check_poll(int fd, int tty) 1122235c7feSXin LI { 113d713e089SXin LI struct pollfd poller[2] = { { fd, POLLIN, 0 }, { tty, POLLIN, 0 } }; 114d713e089SXin LI int timeout = (waiting_for_data && !(scanning_eof && follow_mode == FOLLOW_NAME)) ? -1 : waiting_for_data_delay; 115f80a33eaSXin LI if (!any_data) 116f80a33eaSXin LI { 117f80a33eaSXin LI /* 118f80a33eaSXin LI * Don't do polling if no data has yet been received, 119f80a33eaSXin LI * to allow a program piping data into less to have temporary 120f80a33eaSXin LI * access to the tty (like sudo asking for a password). 121f80a33eaSXin LI */ 122f80a33eaSXin LI return (0); 123f80a33eaSXin LI } 124d713e089SXin LI poll(poller, 2, timeout); 125d713e089SXin LI #if LESSTEST 126*c77c4889SXin LI if (!is_lesstest()) /* Check for ^X only on a real tty. */ 127d713e089SXin LI #endif /*LESSTEST*/ 128d713e089SXin LI { 129d713e089SXin LI if (poller[1].revents & POLLIN) 130d713e089SXin LI { 131*c77c4889SXin LI int ch = getchr(); 132*c77c4889SXin LI if (ch < 0 || ch == intr_char) 133d713e089SXin LI /* Break out of "waiting for data". */ 134d713e089SXin LI return (READ_INTR); 135*c77c4889SXin LI ungetcc_back((char) ch); 1362235c7feSXin LI } 137d713e089SXin LI } 138d713e089SXin LI if (ignore_eoi && exit_F_on_close && (poller[0].revents & (POLLHUP|POLLIN)) == POLLHUP) 139d713e089SXin LI /* Break out of F loop on HUP due to --exit-follow-on-close. */ 140d713e089SXin LI return (READ_INTR); 141d713e089SXin LI if ((poller[0].revents & (POLLIN|POLLHUP|POLLERR)) == 0) 142d713e089SXin LI /* No data available; let caller take action, then try again. */ 143d713e089SXin LI return (READ_AGAIN); 144d713e089SXin LI /* There is data (or HUP/ERR) available. Safe to call read() without blocking. */ 145d713e089SXin LI return (0); 146d713e089SXin LI } 147d713e089SXin LI #endif /* USE_POLL */ 148d713e089SXin LI 149d713e089SXin LI public int supports_ctrl_x(void) 150d713e089SXin LI { 151*c77c4889SXin LI #if MSDOS_COMPILER==WIN32C 152*c77c4889SXin LI return (TRUE); 153*c77c4889SXin LI #else 154d713e089SXin LI #if USE_POLL 155d713e089SXin LI return (use_poll); 156d713e089SXin LI #else 157d713e089SXin LI return (FALSE); 158d713e089SXin LI #endif /* USE_POLL */ 159*c77c4889SXin LI #endif /* MSDOS_COMPILER==WIN32C */ 160d713e089SXin LI } 161a5f0fb15SPaul Saab 162a5f0fb15SPaul Saab /* 163a5f0fb15SPaul Saab * Like read() system call, but is deliberately interruptible. 164a5f0fb15SPaul Saab * A call to intread() from a signal handler will interrupt 165a5f0fb15SPaul Saab * any pending iread(). 166a5f0fb15SPaul Saab */ 167*c77c4889SXin LI public ssize_t iread(int fd, unsigned char *buf, size_t len) 168a5f0fb15SPaul Saab { 169*c77c4889SXin LI ssize_t n; 170a5f0fb15SPaul Saab 1717f074f9cSXin LI start: 172a5f0fb15SPaul Saab #if MSDOS_COMPILER==WIN32C 173a5f0fb15SPaul Saab if (ABORT_SIGS()) 174a5f0fb15SPaul Saab return (READ_INTR); 175a5f0fb15SPaul Saab #else 176a5f0fb15SPaul Saab #if MSDOS_COMPILER && MSDOS_COMPILER != DJGPPC 177a5f0fb15SPaul Saab if (kbhit()) 178a5f0fb15SPaul Saab { 179a5f0fb15SPaul Saab int c; 180a5f0fb15SPaul Saab 181a5f0fb15SPaul Saab c = getch(); 182a5f0fb15SPaul Saab if (c == '\003') 183a5f0fb15SPaul Saab return (READ_INTR); 184a5f0fb15SPaul Saab ungetch(c); 185a5f0fb15SPaul Saab } 186a5f0fb15SPaul Saab #endif 187a5f0fb15SPaul Saab #endif 188d713e089SXin LI if (!reading && SET_JUMP(read_label)) 189a5f0fb15SPaul Saab { 190a5f0fb15SPaul Saab /* 191a5f0fb15SPaul Saab * We jumped here from intread. 192a5f0fb15SPaul Saab */ 193*c77c4889SXin LI reading = FALSE; 1948ed69c6fSPaul Saab #if HAVE_SIGPROCMASK 1958ed69c6fSPaul Saab { 1968ed69c6fSPaul Saab sigset_t mask; 1978ed69c6fSPaul Saab sigemptyset(&mask); 1988ed69c6fSPaul Saab sigprocmask(SIG_SETMASK, &mask, NULL); 1998ed69c6fSPaul Saab } 2008ed69c6fSPaul Saab #else 201a5f0fb15SPaul Saab #if HAVE_SIGSETMASK 202a5f0fb15SPaul Saab sigsetmask(0); 203a5f0fb15SPaul Saab #else 204a5f0fb15SPaul Saab #ifdef _OSK 205a5f0fb15SPaul Saab sigmask(~0); 206a5f0fb15SPaul Saab #endif 207a5f0fb15SPaul Saab #endif 2088ed69c6fSPaul Saab #endif 209f80a33eaSXin LI #if !MSDOS_COMPILER 210f80a33eaSXin LI if (fd != tty && !ABORT_SIGS()) 211f80a33eaSXin LI /* Non-interrupt signal like SIGWINCH. */ 212f80a33eaSXin LI return (READ_AGAIN); 213f80a33eaSXin LI #endif 214a5f0fb15SPaul Saab return (READ_INTR); 215a5f0fb15SPaul Saab } 216a5f0fb15SPaul Saab 217a5f0fb15SPaul Saab flush(); 218*c77c4889SXin LI reading = TRUE; 219a5f0fb15SPaul Saab #if MSDOS_COMPILER==DJGPPC 220a5f0fb15SPaul Saab if (isatty(fd)) 221a5f0fb15SPaul Saab { 222a5f0fb15SPaul Saab /* 223a5f0fb15SPaul Saab * Don't try reading from a TTY until a character is 224a5f0fb15SPaul Saab * available, because that makes some background programs 225a5f0fb15SPaul Saab * believe DOS is busy in a way that prevents those 226a5f0fb15SPaul Saab * programs from working while "less" waits. 227d713e089SXin LI * {{ This code was added 12 Jan 2007; still needed? }} 228a5f0fb15SPaul Saab */ 229a5f0fb15SPaul Saab fd_set readfds; 230a5f0fb15SPaul Saab 231a5f0fb15SPaul Saab FD_ZERO(&readfds); 232a5f0fb15SPaul Saab FD_SET(fd, &readfds); 233a5f0fb15SPaul Saab if (select(fd+1, &readfds, 0, 0, 0) == -1) 23495270f73SXin LI { 235*c77c4889SXin LI reading = FALSE; 236d713e089SXin LI return (READ_ERR); 237a5f0fb15SPaul Saab } 23895270f73SXin LI } 239a5f0fb15SPaul Saab #endif 2402235c7feSXin LI #if USE_POLL 241*c77c4889SXin LI if (is_tty && fd != tty && use_poll) 2422235c7feSXin LI { 243d713e089SXin LI int ret = check_poll(fd, tty); 244d713e089SXin LI if (ret != 0) 2452235c7feSXin LI { 246d713e089SXin LI if (ret == READ_INTR) 2472235c7feSXin LI sigs |= S_INTERRUPT; 248*c77c4889SXin LI reading = FALSE; 249d713e089SXin LI return (ret); 2502235c7feSXin LI } 2512235c7feSXin LI } 2522235c7feSXin LI #else 2532235c7feSXin LI #if MSDOS_COMPILER==WIN32C 254f80a33eaSXin LI if (win32_kbhit()) 255f80a33eaSXin LI { 256f80a33eaSXin LI int c; 257f80a33eaSXin LI 258f80a33eaSXin LI c = WIN32getch(); 259f80a33eaSXin LI if (c == intr_char) 2602235c7feSXin LI { 2612235c7feSXin LI sigs |= S_INTERRUPT; 262*c77c4889SXin LI reading = FALSE; 2632235c7feSXin LI return (READ_INTR); 2642235c7feSXin LI } 265f80a33eaSXin LI WIN32ungetch(c); 266f80a33eaSXin LI } 2672235c7feSXin LI #endif 2682235c7feSXin LI #endif 269a5f0fb15SPaul Saab n = read(fd, buf, len); 270*c77c4889SXin LI reading = FALSE; 271a5f0fb15SPaul Saab #if 1 272a5f0fb15SPaul Saab /* 273a5f0fb15SPaul Saab * This is a kludge to workaround a problem on some systems 274a5f0fb15SPaul Saab * where terminating a remote tty connection causes read() to 275a5f0fb15SPaul Saab * start returning 0 forever, instead of -1. 276a5f0fb15SPaul Saab */ 277a5f0fb15SPaul Saab { 278a5f0fb15SPaul Saab if (!ignore_eoi) 279a5f0fb15SPaul Saab { 280a5f0fb15SPaul Saab if (n == 0) 281a5f0fb15SPaul Saab consecutive_nulls++; 282a5f0fb15SPaul Saab else 283a5f0fb15SPaul Saab consecutive_nulls = 0; 284a5f0fb15SPaul Saab if (consecutive_nulls > 20) 285a5f0fb15SPaul Saab quit(QUIT_ERROR); 286a5f0fb15SPaul Saab } 287a5f0fb15SPaul Saab } 288a5f0fb15SPaul Saab #endif 289a5f0fb15SPaul Saab if (n < 0) 2907f074f9cSXin LI { 2917f074f9cSXin LI #if HAVE_ERRNO 2927f074f9cSXin LI /* 2937f074f9cSXin LI * Certain values of errno indicate we should just retry the read. 2947f074f9cSXin LI */ 2957f074f9cSXin LI #if MUST_DEFINE_ERRNO 2967f074f9cSXin LI extern int errno; 2977f074f9cSXin LI #endif 2987f074f9cSXin LI #ifdef EINTR 2997f074f9cSXin LI if (errno == EINTR) 3007f074f9cSXin LI goto start; 3017f074f9cSXin LI #endif 3027f074f9cSXin LI #ifdef EAGAIN 3037f074f9cSXin LI if (errno == EAGAIN) 3047f074f9cSXin LI goto start; 3057f074f9cSXin LI #endif 3067f074f9cSXin LI #endif 307d713e089SXin LI return (READ_ERR); 3087f074f9cSXin LI } 309f80a33eaSXin LI #if USE_POLL 310f80a33eaSXin LI if (fd != tty && n > 0) 311f80a33eaSXin LI any_data = TRUE; 312f80a33eaSXin LI #endif 313a5f0fb15SPaul Saab return (n); 314a5f0fb15SPaul Saab } 315a5f0fb15SPaul Saab 316a5f0fb15SPaul Saab /* 317a5f0fb15SPaul Saab * Interrupt a pending iread(). 318a5f0fb15SPaul Saab */ 319d713e089SXin LI public void intread(void) 320a5f0fb15SPaul Saab { 321a5f0fb15SPaul Saab LONG_JUMP(read_label, 1); 322a5f0fb15SPaul Saab } 323a5f0fb15SPaul Saab 324a5f0fb15SPaul Saab /* 325a5f0fb15SPaul Saab * Return the current time. 326a5f0fb15SPaul Saab */ 327a5f0fb15SPaul Saab #if HAVE_TIME 328d713e089SXin LI public time_type get_time(void) 329a5f0fb15SPaul Saab { 330a5f0fb15SPaul Saab time_type t; 331a5f0fb15SPaul Saab 332a5f0fb15SPaul Saab time(&t); 333a5f0fb15SPaul Saab return (t); 334a5f0fb15SPaul Saab } 335a5f0fb15SPaul Saab #endif 336a5f0fb15SPaul Saab 337a5f0fb15SPaul Saab 338a5f0fb15SPaul Saab #if !HAVE_STRERROR 339a5f0fb15SPaul Saab /* 340a5f0fb15SPaul Saab * Local version of strerror, if not available from the system. 341a5f0fb15SPaul Saab */ 342d713e089SXin LI static char * strerror(int err) 343a5f0fb15SPaul Saab { 34495270f73SXin LI static char buf[INT_STRLEN_BOUND(int)+12]; 3452235c7feSXin LI #if HAVE_SYS_ERRLIST 346a5f0fb15SPaul Saab extern char *sys_errlist[]; 347a5f0fb15SPaul Saab extern int sys_nerr; 348a5f0fb15SPaul Saab 349a5f0fb15SPaul Saab if (err < sys_nerr) 350a5f0fb15SPaul Saab return sys_errlist[err]; 3512235c7feSXin LI #endif 352a5f0fb15SPaul Saab sprintf(buf, "Error %d", err); 353a5f0fb15SPaul Saab return buf; 354a5f0fb15SPaul Saab } 355a5f0fb15SPaul Saab #endif 356a5f0fb15SPaul Saab 357a5f0fb15SPaul Saab /* 358a5f0fb15SPaul Saab * errno_message: Return an error message based on the value of "errno". 359a5f0fb15SPaul Saab */ 360*c77c4889SXin LI public char * errno_message(constant char *filename) 361a5f0fb15SPaul Saab { 3621ea31627SRobert Watson char *p; 3631ea31627SRobert Watson char *m; 364*c77c4889SXin LI size_t len; 365a5f0fb15SPaul Saab #if HAVE_ERRNO 366a5f0fb15SPaul Saab #if MUST_DEFINE_ERRNO 367a5f0fb15SPaul Saab extern int errno; 368a5f0fb15SPaul Saab #endif 369a5f0fb15SPaul Saab p = strerror(errno); 370a5f0fb15SPaul Saab #else 371a5f0fb15SPaul Saab p = "cannot open"; 372a5f0fb15SPaul Saab #endif 373*c77c4889SXin LI len = strlen(filename) + strlen(p) + 3; 3746dcb072bSXin LI m = (char *) ecalloc(len, sizeof(char)); 3756dcb072bSXin LI SNPRINTF2(m, len, "%s: %s", filename, p); 376a5f0fb15SPaul Saab return (m); 377a5f0fb15SPaul Saab } 378a5f0fb15SPaul Saab 379d713e089SXin LI /* 380d713e089SXin LI * Return a description of a signal. 381d713e089SXin LI * The return value is good until the next call to this function. 382d713e089SXin LI */ 383*c77c4889SXin LI public constant char * signal_message(int sig) 38433096f16SXin LI { 385d713e089SXin LI static char sigbuf[sizeof("Signal ") + INT_STRLEN_BOUND(sig) + 1]; 386d713e089SXin LI #if HAVE_STRSIGNAL 387*c77c4889SXin LI constant char *description = strsignal(sig); 388d713e089SXin LI if (description) 389d713e089SXin LI return description; 39033096f16SXin LI #endif 391d713e089SXin LI sprintf(sigbuf, "Signal %d", sig); 392d713e089SXin LI return sigbuf; 393d713e089SXin LI } 394d713e089SXin LI 395d713e089SXin LI /* 396d713e089SXin LI * Return (VAL * NUM) / DEN, where DEN is positive 397d713e089SXin LI * and min(VAL, NUM) <= DEN so the result cannot overflow. 398d713e089SXin LI * Round to the nearest integer, breaking ties by rounding to even. 399d713e089SXin LI */ 400*c77c4889SXin LI public uintmax umuldiv(uintmax val, uintmax num, uintmax den) 401d713e089SXin LI { 402d713e089SXin LI /* 403d713e089SXin LI * Like round(val * (double) num / den), but without rounding error. 404d713e089SXin LI * Overflow cannot occur, so there is no need for floating point. 405d713e089SXin LI */ 406d713e089SXin LI uintmax q = val / den; 407d713e089SXin LI uintmax r = val % den; 408d713e089SXin LI uintmax qnum = q * num; 409d713e089SXin LI uintmax rnum = r * num; 410d713e089SXin LI uintmax quot = qnum + rnum / den; 411d713e089SXin LI uintmax rem = rnum % den; 412d713e089SXin LI return quot + (den / 2 < rem + (quot & ~den & 1)); 41333096f16SXin LI } 41433096f16SXin LI 415a5f0fb15SPaul Saab /* 416a5f0fb15SPaul Saab * Return the ratio of two POSITIONS, as a percentage. 417a5f0fb15SPaul Saab * {{ Assumes a POSITION is a long int. }} 418a5f0fb15SPaul Saab */ 419d713e089SXin LI public int percentage(POSITION num, POSITION den) 420a5f0fb15SPaul Saab { 421*c77c4889SXin LI return (int) muldiv(num, 100, den); 422a5f0fb15SPaul Saab } 423a5f0fb15SPaul Saab 424a5f0fb15SPaul Saab /* 425a5f0fb15SPaul Saab * Return the specified percentage of a POSITION. 426d713e089SXin LI * Assume (0 <= POS && 0 <= PERCENT <= 100 427d713e089SXin LI * && 0 <= FRACTION < (PERCENT == 100 ? 1 : NUM_FRAC_DENOM)), 428d713e089SXin LI * so the result cannot overflow. Round to even. 429a5f0fb15SPaul Saab */ 430d713e089SXin LI public POSITION percent_pos(POSITION pos, int percent, long fraction) 431a5f0fb15SPaul Saab { 432d713e089SXin LI /* 433d713e089SXin LI * Change from percent (parts per 100) 434d713e089SXin LI * to pctden (parts per 100 * NUM_FRAC_DENOM). 435d713e089SXin LI */ 436d713e089SXin LI POSITION pctden = (percent * NUM_FRAC_DENOM) + fraction; 437c9346414SPaul Saab 438*c77c4889SXin LI return (POSITION) muldiv(pos, pctden, 100 * NUM_FRAC_DENOM); 439a5f0fb15SPaul Saab } 440a5f0fb15SPaul Saab 441c9346414SPaul Saab #if !HAVE_STRCHR 442c9346414SPaul Saab /* 443c9346414SPaul Saab * strchr is used by regexp.c. 444c9346414SPaul Saab */ 445d713e089SXin LI char * strchr(char *s, char c) 446c9346414SPaul Saab { 447c9346414SPaul Saab for ( ; *s != '\0'; s++) 448c9346414SPaul Saab if (*s == c) 449c9346414SPaul Saab return (s); 450c9346414SPaul Saab if (c == '\0') 451c9346414SPaul Saab return (s); 452c9346414SPaul Saab return (NULL); 453c9346414SPaul Saab } 454c9346414SPaul Saab #endif 455c9346414SPaul Saab 456c9346414SPaul Saab #if !HAVE_MEMCPY 457*c77c4889SXin LI void * memcpy(void *dst, void *src, size_t len) 458c9346414SPaul Saab { 459c9346414SPaul Saab char *dstp = (char *) dst; 460c9346414SPaul Saab char *srcp = (char *) src; 461c9346414SPaul Saab int i; 462c9346414SPaul Saab 463c9346414SPaul Saab for (i = 0; i < len; i++) 464c9346414SPaul Saab dstp[i] = srcp[i]; 465c9346414SPaul Saab return (dst); 466c9346414SPaul Saab } 467c9346414SPaul Saab #endif 468c9346414SPaul Saab 469a5f0fb15SPaul Saab #ifdef _OSK_MWC32 470a5f0fb15SPaul Saab 471a5f0fb15SPaul Saab /* 472a5f0fb15SPaul Saab * This implements an ANSI-style intercept setup for Microware C 3.2 473a5f0fb15SPaul Saab */ 474d713e089SXin LI public int os9_signal(int type, RETSIGTYPE (*handler)()) 475a5f0fb15SPaul Saab { 476a5f0fb15SPaul Saab intercept(handler); 477a5f0fb15SPaul Saab } 478a5f0fb15SPaul Saab 479a5f0fb15SPaul Saab #include <sgstat.h> 480a5f0fb15SPaul Saab 481d713e089SXin LI int isatty(int f) 482a5f0fb15SPaul Saab { 483a5f0fb15SPaul Saab struct sgbuf sgbuf; 484a5f0fb15SPaul Saab 485a5f0fb15SPaul Saab if (_gs_opt(f, &sgbuf) < 0) 486a5f0fb15SPaul Saab return -1; 487a5f0fb15SPaul Saab return (sgbuf.sg_class == 0); 488a5f0fb15SPaul Saab } 489a5f0fb15SPaul Saab 490a5f0fb15SPaul Saab #endif 4912235c7feSXin LI 492d713e089SXin LI public void sleep_ms(int ms) 4932235c7feSXin LI { 4942235c7feSXin LI #if MSDOS_COMPILER==WIN32C 4952235c7feSXin LI Sleep(ms); 4962235c7feSXin LI #else 4972235c7feSXin LI #if HAVE_NANOSLEEP 4982235c7feSXin LI int sec = ms / 1000; 4992235c7feSXin LI struct timespec t = { sec, (ms - sec*1000) * 1000000 }; 5002235c7feSXin LI nanosleep(&t, NULL); 5012235c7feSXin LI #else 5022235c7feSXin LI #if HAVE_USLEEP 503*c77c4889SXin LI usleep(ms * 1000); 5042235c7feSXin LI #else 505d713e089SXin LI sleep(ms / 1000 + (ms % 1000 != 0)); 5062235c7feSXin LI #endif 5072235c7feSXin LI #endif 5082235c7feSXin LI #endif 5092235c7feSXin LI } 510