xref: /freebsd-src/contrib/less/os.c (revision c77c488926555ca344ae3a417544cf7a720e1de1)
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