xref: /openbsd/bin/csh/file.c (revision 157d47b0)
1*157d47b0Sderaadt /*	$OpenBSD: file.c,v 1.40 2020/10/06 01:40:43 deraadt Exp $	*/
226b220adSmillert /*	$NetBSD: file.c,v 1.11 1996/11/08 19:34:37 christos Exp $	*/
3df930be7Sderaadt 
4df930be7Sderaadt /*-
5df930be7Sderaadt  * Copyright (c) 1980, 1991, 1993
6df930be7Sderaadt  *	The Regents of the University of California.  All rights reserved.
7df930be7Sderaadt  *
8df930be7Sderaadt  * Redistribution and use in source and binary forms, with or without
9df930be7Sderaadt  * modification, are permitted provided that the following conditions
10df930be7Sderaadt  * are met:
11df930be7Sderaadt  * 1. Redistributions of source code must retain the above copyright
12df930be7Sderaadt  *    notice, this list of conditions and the following disclaimer.
13df930be7Sderaadt  * 2. Redistributions in binary form must reproduce the above copyright
14df930be7Sderaadt  *    notice, this list of conditions and the following disclaimer in the
15df930be7Sderaadt  *    documentation and/or other materials provided with the distribution.
1629295d1cSmillert  * 3. Neither the name of the University nor the names of its contributors
17df930be7Sderaadt  *    may be used to endorse or promote products derived from this software
18df930be7Sderaadt  *    without specific prior written permission.
19df930be7Sderaadt  *
20df930be7Sderaadt  * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND
21df930be7Sderaadt  * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
22df930be7Sderaadt  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
23df930be7Sderaadt  * ARE DISCLAIMED.  IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE
24df930be7Sderaadt  * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
25df930be7Sderaadt  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
26df930be7Sderaadt  * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
27df930be7Sderaadt  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
28df930be7Sderaadt  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
29df930be7Sderaadt  * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
30df930be7Sderaadt  * SUCH DAMAGE.
31df930be7Sderaadt  */
32df930be7Sderaadt 
33df930be7Sderaadt #include <sys/ioctl.h>
34df930be7Sderaadt #include <sys/stat.h>
359448619aSanton #include <sys/types.h>
369448619aSanton 
37df930be7Sderaadt #include <dirent.h>
38c0d918cdSanton #include <errno.h>
399448619aSanton #include <limits.h>
40df930be7Sderaadt #include <pwd.h>
41df930be7Sderaadt #include <stdlib.h>
42c0d918cdSanton #include <string.h>
439448619aSanton #include <termios.h>
44df930be7Sderaadt #include <unistd.h>
45df930be7Sderaadt 
46df930be7Sderaadt #include "csh.h"
47df930be7Sderaadt #include "extern.h"
48df930be7Sderaadt 
49df930be7Sderaadt /*
50df930be7Sderaadt  * Tenex style file name recognition, .. and more.
51df930be7Sderaadt  * History:
52df930be7Sderaadt  *	Author: Ken Greer, Sept. 1975, CMU.
53df930be7Sderaadt  *	Finally got around to adding to the Cshell., Ken Greer, Dec. 1981.
54df930be7Sderaadt  */
55df930be7Sderaadt 
56df930be7Sderaadt #ifndef TRUE
57df930be7Sderaadt #define TRUE 1
58df930be7Sderaadt #endif
59df930be7Sderaadt #ifndef FALSE
60df930be7Sderaadt #define FALSE 0
61df930be7Sderaadt #endif
62df930be7Sderaadt 
63df930be7Sderaadt #define	ESC		'\033'
64c0d918cdSanton #define	TABWIDTH	8
65c0d918cdSanton 
66df930be7Sderaadt typedef enum {
679448619aSanton 	LIST,
689448619aSanton 	RECOGNIZE
69df930be7Sderaadt } COMMAND;
70df930be7Sderaadt 
71c0d918cdSanton struct cmdline {
72c0d918cdSanton 	int	 fdin;
73c0d918cdSanton 	int	 fdout;
74*157d47b0Sderaadt 	int	 istty;
75c0d918cdSanton 	int	 flags;
76c0d918cdSanton #define	CL_ALTWERASE	0x1
77c0d918cdSanton #define	CL_PROMPT	0x2
78c0d918cdSanton 	char	*buf;
79c0d918cdSanton 	size_t	 len;
80c0d918cdSanton 	size_t	 size;
81c0d918cdSanton 	size_t	 cursor;
82c0d918cdSanton };
83c0d918cdSanton 
84c0d918cdSanton /* Command line auxiliary functions. */
85c0d918cdSanton static void	 cl_beep(struct cmdline *);
86c0d918cdSanton static void	 cl_flush(struct cmdline *);
87c0d918cdSanton static int	 cl_getc(struct cmdline *);
88c0d918cdSanton static Char	*cl_lastw(struct cmdline *);
89c0d918cdSanton static void	 cl_putc(struct cmdline *, int);
90c0d918cdSanton static void	 cl_visc(struct cmdline *, int);
91c0d918cdSanton 
92c0d918cdSanton /* Command line editing functions. */
93c0d918cdSanton static int	cl_abort(struct cmdline *, int);
94c0d918cdSanton static int	cl_erasec(struct cmdline *, int);
95c0d918cdSanton static int	cl_erasew(struct cmdline *, int);
96c0d918cdSanton static int	cl_insert(struct cmdline *, int);
97c0d918cdSanton static int	cl_kill(struct cmdline *, int);
98c0d918cdSanton static int	cl_list(struct cmdline *, int);
99c0d918cdSanton static int	cl_literal(struct cmdline *, int);
100c0d918cdSanton static int	cl_recognize(struct cmdline *, int);
101c0d918cdSanton static int	cl_reprint(struct cmdline *, int);
102c0d918cdSanton static int	cl_status(struct cmdline *, int);
103c0d918cdSanton 
104c0d918cdSanton static const struct termios	*setup_tty(int);
105c0d918cdSanton 
106c72b5b24Smillert static void	 catn(Char *, Char *, int);
107c72b5b24Smillert static void	 copyn(Char *, Char *, int);
108c72b5b24Smillert static Char	 filetype(Char *, Char *);
109c72b5b24Smillert static void	 print_by_column(Char *, Char *[], int);
110c72b5b24Smillert static Char	*tilde(Char *, Char *);
111c72b5b24Smillert static void	 extract_dir_and_name(Char *, Char *, Char *);
112c72b5b24Smillert static Char	*getentry(DIR *, int);
113b1f6b197Smillert static void	 free_items(Char **, int);
114c72b5b24Smillert static int	 tsearch(Char *, COMMAND, int);
115c72b5b24Smillert static int	 recognize(Char *, Char *, int, int);
116c72b5b24Smillert static int	 is_prefix(Char *, Char *);
117c72b5b24Smillert static int	 is_suffix(Char *, Char *);
118c72b5b24Smillert static int	 ignored(Char *);
119df930be7Sderaadt 
120df930be7Sderaadt /*
121df930be7Sderaadt  * Put this here so the binary can be patched with adb to enable file
122df930be7Sderaadt  * completion by default.  Filec controls completion, nobeep controls
123df930be7Sderaadt  * ringing the terminal bell on incomplete expansions.
124df930be7Sderaadt  */
125df930be7Sderaadt bool    filec = 0;
126df930be7Sderaadt 
127df930be7Sderaadt static void
cl_flush(struct cmdline * cl)128c0d918cdSanton cl_flush(struct cmdline *cl)
129c0d918cdSanton {
130c0d918cdSanton 	size_t	i, len;
131c0d918cdSanton 	int	c;
132c0d918cdSanton 
133c0d918cdSanton 	if (cl->flags & CL_PROMPT) {
134c0d918cdSanton 		cl->flags &= ~CL_PROMPT;
135c0d918cdSanton 		printprompt();
136c0d918cdSanton 	}
137c0d918cdSanton 
138c0d918cdSanton 	if (cl->cursor < cl->len) {
139c0d918cdSanton 		for (; cl->cursor < cl->len; cl->cursor++)
140c0d918cdSanton 			cl_visc(cl, cl->buf[cl->cursor]);
141c0d918cdSanton 	} else if (cl->cursor > cl->len) {
142c0d918cdSanton 		len = cl->cursor - cl->len;
143c0d918cdSanton 		for (i = len; i > 0; i--) {
144c0d918cdSanton 			c = cl->buf[--cl->cursor];
145c0d918cdSanton 			if (c == '\t')
146c0d918cdSanton 				len += TABWIDTH - 1;
147c0d918cdSanton 			else if (iscntrl(c))
148c0d918cdSanton 				len++;	/* account for leading ^ */
149c0d918cdSanton 		}
150c0d918cdSanton 		for (i = 0; i < len; i++)
151c0d918cdSanton 			cl_putc(cl, '\b');
152c0d918cdSanton 		for (i = 0; i < len; i++)
153c0d918cdSanton 			cl_putc(cl, ' ');
154c0d918cdSanton 		for (i = 0; i < len; i++)
155c0d918cdSanton 			cl_putc(cl, '\b');
156c0d918cdSanton 		cl->cursor = cl->len;
157c0d918cdSanton 	}
158c0d918cdSanton }
159c0d918cdSanton 
160c0d918cdSanton static int
cl_getc(struct cmdline * cl)161c0d918cdSanton cl_getc(struct cmdline *cl)
162c0d918cdSanton {
163c0d918cdSanton 	ssize_t		n;
164c51b6a6dSderaadt 	unsigned char	c;
165c0d918cdSanton 
166c0d918cdSanton 	for (;;) {
167c0d918cdSanton 		n = read(cl->fdin, &c, 1);
168c0d918cdSanton 		switch (n) {
169c0d918cdSanton 		case -1:
170c0d918cdSanton 			if (errno == EINTR)
171c0d918cdSanton 				continue;
172c0d918cdSanton 			/* FALLTHROUGH */
173c0d918cdSanton 		case 0:
174c0d918cdSanton 			return 0;
175c0d918cdSanton 		default:
176c0d918cdSanton 			return c & 0x7F;
177c0d918cdSanton 		}
178c0d918cdSanton 	}
179c0d918cdSanton }
180c0d918cdSanton 
181c0d918cdSanton static Char *
cl_lastw(struct cmdline * cl)182c0d918cdSanton cl_lastw(struct cmdline *cl)
183c0d918cdSanton {
184c0d918cdSanton 	static Char		 word[BUFSIZ];
185c51b6a6dSderaadt 	const unsigned char	*delimiters = " '\"\t;&<>()|^%";
186c0d918cdSanton 	Char			*cp;
187c0d918cdSanton 	size_t			 i;
188c0d918cdSanton 
189c0d918cdSanton 	for (i = cl->len; i > 0; i--)
190c0d918cdSanton 		if (strchr(delimiters, cl->buf[i - 1]) != NULL)
191c0d918cdSanton 			break;
192c0d918cdSanton 
193c0d918cdSanton 	cp = word;
194c0d918cdSanton 	for (; i < cl->len; i++)
195c0d918cdSanton 		*cp++ = cl->buf[i];
196c0d918cdSanton 	*cp = '\0';
197c0d918cdSanton 
198c0d918cdSanton 	return word;
199c0d918cdSanton }
200c0d918cdSanton 
201c0d918cdSanton static void
cl_putc(struct cmdline * cl,int c)20221aa5442Santon cl_putc(struct cmdline *cl, int c)
203c0d918cdSanton {
20421aa5442Santon 	unsigned char	cc = c;
205c51b6a6dSderaadt 
20621aa5442Santon 	write(cl->fdout, &cc, 1);
207c0d918cdSanton }
208c0d918cdSanton 
209c0d918cdSanton static void
cl_visc(struct cmdline * cl,int c)210c0d918cdSanton cl_visc(struct cmdline *cl, int c)
211c0d918cdSanton {
212c0d918cdSanton #define	UNCNTRL(x)	((x) == 0x7F ? '?' : ((x) | 0x40))
213c0d918cdSanton 	int	i;
214c0d918cdSanton 
215c0d918cdSanton 	if (c == '\t') {
216c0d918cdSanton 		for (i = 0; i < TABWIDTH; i++)
217c0d918cdSanton 			cl_putc(cl, ' ');
218c0d918cdSanton 	} else if (c != '\n' && iscntrl(c)) {
219c0d918cdSanton 		cl_putc(cl, '^');
220c0d918cdSanton 		cl_putc(cl, UNCNTRL(c));
221c0d918cdSanton 	} else {
222c0d918cdSanton 		cl_putc(cl, c);
223c0d918cdSanton 	}
224c0d918cdSanton }
225c0d918cdSanton 
226c0d918cdSanton static int
cl_abort(struct cmdline * cl,int c)227c0d918cdSanton cl_abort(struct cmdline *cl, int c)
228c0d918cdSanton {
229c0d918cdSanton 	cl_visc(cl, c);
230046eeb90Santon 
231046eeb90Santon 	/* Abort while/foreach loop prematurely. */
232d8e3aed7Santon 	if (whyles) {
233*157d47b0Sderaadt 		if (cl->istty)
234d8e3aed7Santon 			setup_tty(0);
235046eeb90Santon 		kill(getpid(), SIGINT);
236d8e3aed7Santon 	}
237046eeb90Santon 
238c0d918cdSanton 	cl_putc(cl, '\n');
239c0d918cdSanton 	cl->len = cl->cursor = 0;
240c0d918cdSanton 	cl->flags |= CL_PROMPT;
241c0d918cdSanton 
242c0d918cdSanton 	return 0;
243c0d918cdSanton }
244c0d918cdSanton 
245c0d918cdSanton static int
cl_erasec(struct cmdline * cl,int c)246c0d918cdSanton cl_erasec(struct cmdline *cl, int c)
247c0d918cdSanton {
248c0d918cdSanton 	if (cl->len > 0)
249c0d918cdSanton 		cl->len--;
250c0d918cdSanton 
251c0d918cdSanton 	return 0;
252c0d918cdSanton }
253c0d918cdSanton 
254c0d918cdSanton static int
cl_erasew(struct cmdline * cl,int c)255c0d918cdSanton cl_erasew(struct cmdline *cl, int c)
256c0d918cdSanton {
257c51b6a6dSderaadt 	const unsigned char	*ws = " \t";
258c0d918cdSanton 
259c0d918cdSanton 	for (; cl->len > 0; cl->len--)
260d417edc4Santon 		if (strchr(ws, cl->buf[cl->len - 1]) == NULL &&
261d417edc4Santon 		    ((cl->flags & CL_ALTWERASE) == 0 ||
262d417edc4Santon 		     isalpha(cl->buf[cl->len - 1])))
263c0d918cdSanton 			break;
264c0d918cdSanton 	for (; cl->len > 0; cl->len--)
265d417edc4Santon 		if (strchr(ws, cl->buf[cl->len - 1]) != NULL ||
266d417edc4Santon 		    ((cl->flags & CL_ALTWERASE) &&
267d417edc4Santon 		     !isalpha(cl->buf[cl->len - 1])))
268c0d918cdSanton 			break;
269c0d918cdSanton 
270c0d918cdSanton 	return 0;
271c0d918cdSanton }
272c0d918cdSanton 
273c0d918cdSanton static void
cl_beep(struct cmdline * cl)274c0d918cdSanton cl_beep(struct cmdline *cl)
275c0d918cdSanton {
276c0d918cdSanton 	if (adrof(STRnobeep) == 0)
277c0d918cdSanton 		cl_putc(cl, '\007');
278c0d918cdSanton }
279c0d918cdSanton 
280c0d918cdSanton static int
cl_insert(struct cmdline * cl,int c)281c0d918cdSanton cl_insert(struct cmdline *cl, int c)
282c0d918cdSanton {
283c0d918cdSanton 	if (cl->len == cl->size)
284c0d918cdSanton 		return 1;
285c0d918cdSanton 
286c0d918cdSanton 	cl->buf[cl->len++] = c;
287c0d918cdSanton 
288c0d918cdSanton 	if (c == '\n')
289c0d918cdSanton 		return 1;
290c0d918cdSanton 
291c0d918cdSanton 	return 0;
292c0d918cdSanton }
293c0d918cdSanton 
294c0d918cdSanton static int
cl_kill(struct cmdline * cl,int c)295c0d918cdSanton cl_kill(struct cmdline *cl, int c)
296c0d918cdSanton {
297c0d918cdSanton 	cl->len = 0;
298c0d918cdSanton 
299c0d918cdSanton 	return 0;
300c0d918cdSanton }
301c0d918cdSanton 
302c0d918cdSanton static int
cl_list(struct cmdline * cl,int c)303c0d918cdSanton cl_list(struct cmdline *cl, int c)
304c0d918cdSanton {
305c0d918cdSanton 	Char	*word;
306c0d918cdSanton 	size_t	 len;
307c0d918cdSanton 
308c0d918cdSanton 	if (adrof(STRignoreeof) || cl->len > 0)
309c0d918cdSanton 		cl_visc(cl, c);
310c0d918cdSanton 
311c0d918cdSanton 	if (cl->len == 0)
312c0d918cdSanton 		return 1;
313c0d918cdSanton 
314c0d918cdSanton 	cl_putc(cl, '\n');
315c0d918cdSanton 	cl->cursor = 0;
316c0d918cdSanton 	cl->flags |= CL_PROMPT;
317c0d918cdSanton 
318c0d918cdSanton 	word = cl_lastw(cl);
319c0d918cdSanton 	len = Strlen(word);
320c0d918cdSanton 	tsearch(word, LIST, BUFSIZ - len - 1);	/* NUL */
321c0d918cdSanton 
322c0d918cdSanton 	return 0;
323c0d918cdSanton }
324c0d918cdSanton 
325c0d918cdSanton static int
cl_literal(struct cmdline * cl,int c)326c0d918cdSanton cl_literal(struct cmdline *cl, int c)
327c0d918cdSanton {
328c0d918cdSanton 	int	literal;
329c0d918cdSanton 
330c0d918cdSanton 	literal = cl_getc(cl);
331c0d918cdSanton 	if (literal == '\n')
332c0d918cdSanton 		literal = '\r';
333c0d918cdSanton 	cl_insert(cl, literal);
334c0d918cdSanton 
335c0d918cdSanton 	return 0;
336c0d918cdSanton }
337c0d918cdSanton 
338c0d918cdSanton static int
cl_recognize(struct cmdline * cl,int c)339c0d918cdSanton cl_recognize(struct cmdline *cl, int c)
340c0d918cdSanton {
341c0d918cdSanton 	Char	*word;
342c0d918cdSanton 	size_t	 len;
343c0d918cdSanton 	int	 nitems;
344c0d918cdSanton 
345c0d918cdSanton 	if (cl->len == 0) {
346c0d918cdSanton 		cl_beep(cl);
347c0d918cdSanton 		return 0;
348c0d918cdSanton 	}
349c0d918cdSanton 
350c0d918cdSanton 	word = cl_lastw(cl);
351c0d918cdSanton 	len = Strlen(word);
352c0d918cdSanton 	nitems = tsearch(word, RECOGNIZE, BUFSIZ - len - 1);	/* NUL */
353c0d918cdSanton 	for (word += len; *word != '\0'; word++)
354c0d918cdSanton 		cl_insert(cl, *word);
355c0d918cdSanton 	if (nitems != 1)
356c0d918cdSanton 		cl_beep(cl);
357c0d918cdSanton 
358c0d918cdSanton 	return 0;
359c0d918cdSanton }
360c0d918cdSanton 
361c0d918cdSanton static int
cl_reprint(struct cmdline * cl,int c)362c0d918cdSanton cl_reprint(struct cmdline *cl, int c)
363c0d918cdSanton {
364c0d918cdSanton 	cl_visc(cl, c);
365c0d918cdSanton 	cl_putc(cl, '\n');
366c0d918cdSanton 	cl->cursor = 0;
367c0d918cdSanton 
368c0d918cdSanton 	return 0;
369c0d918cdSanton }
370c0d918cdSanton 
371c0d918cdSanton static int
cl_status(struct cmdline * cl,int c)372c0d918cdSanton cl_status(struct cmdline *cl, int c)
373c0d918cdSanton {
374c0d918cdSanton 	cl->cursor = 0;
375*157d47b0Sderaadt 	if (cl->istty)
3766b4ead63Sderaadt 		ioctl(cl->fdin, TIOCSTAT);
377c0d918cdSanton 
378c0d918cdSanton 	return 0;
379c0d918cdSanton }
380c0d918cdSanton 
381c0d918cdSanton const struct termios *
setup_tty(int on)382e757c91eSderaadt setup_tty(int on)
383df930be7Sderaadt {
384c0d918cdSanton 	static struct termios	newtio, oldtio;
385df930be7Sderaadt 
386df930be7Sderaadt 	if (on) {
387c0d918cdSanton 		tcgetattr(SHIN, &oldtio);
388c0d918cdSanton 
389c0d918cdSanton 		newtio = oldtio;
390c0d918cdSanton 		newtio.c_lflag &= ~(ECHO | ICANON | ISIG);
391c0d918cdSanton 		newtio.c_cc[VEOL] = ESC;
392c0d918cdSanton 		newtio.c_cc[VLNEXT] = _POSIX_VDISABLE;
393c0d918cdSanton 		newtio.c_cc[VMIN] = 1;
394c0d918cdSanton 		newtio.c_cc[VTIME] = 0;
395c0d918cdSanton 	} else {
396c0d918cdSanton 		newtio = oldtio;
397df930be7Sderaadt 	}
398df930be7Sderaadt 
399c0d918cdSanton 	tcsetattr(SHIN, TCSADRAIN, &newtio);
400df930be7Sderaadt 
401df930be7Sderaadt 	/*
402c0d918cdSanton 	 * Since VLNEXT is disabled, restore its previous value in order to make
403c0d918cdSanton 	 * the key detectable.
404df930be7Sderaadt 	 */
405c0d918cdSanton 	newtio.c_cc[VLNEXT] = oldtio.c_cc[VLNEXT];
406df930be7Sderaadt 
407c0d918cdSanton 	return &newtio;
408df930be7Sderaadt }
409df930be7Sderaadt 
410df930be7Sderaadt /*
411df930be7Sderaadt  * Concatenate src onto tail of des.
412df930be7Sderaadt  * Des is a string whose maximum length is count.
413df930be7Sderaadt  * Always null terminate.
414df930be7Sderaadt  */
415df930be7Sderaadt static void
catn(Char * des,Char * src,int count)416e757c91eSderaadt catn(Char *des, Char *src, int count)
417df930be7Sderaadt {
418df930be7Sderaadt     while (--count >= 0 && *des)
419df930be7Sderaadt 	des++;
420df930be7Sderaadt     while (--count >= 0)
421df930be7Sderaadt 	if ((*des++ = *src++) == 0)
422df930be7Sderaadt 	    return;
423df930be7Sderaadt     *des = '\0';
424df930be7Sderaadt }
425df930be7Sderaadt 
426df930be7Sderaadt /*
427aead66f0Sderaadt  * Places Char's like strlcpy, but no special return value.
428df930be7Sderaadt  */
429df930be7Sderaadt static void
copyn(Char * des,Char * src,int count)430e757c91eSderaadt copyn(Char *des, Char *src, int count)
431df930be7Sderaadt {
432df930be7Sderaadt     while (--count >= 0)
433df930be7Sderaadt 	if ((*des++ = *src++) == 0)
434df930be7Sderaadt 	    return;
435df930be7Sderaadt     *des = '\0';
436df930be7Sderaadt }
437df930be7Sderaadt 
438df930be7Sderaadt static  Char
filetype(Char * dir,Char * file)439e757c91eSderaadt filetype(Char *dir, Char *file)
440df930be7Sderaadt {
441b9fc9a72Sderaadt     Char    path[PATH_MAX];
442df930be7Sderaadt     struct stat statb;
443df930be7Sderaadt 
4446a01f4acSderaadt     Strlcpy(path, dir, sizeof path/sizeof(Char));
4456a01f4acSderaadt     catn(path, file, sizeof(path) / sizeof(Char));
446df930be7Sderaadt     if (lstat(short2str(path), &statb) == 0) {
447df930be7Sderaadt 	switch (statb.st_mode & S_IFMT) {
448df930be7Sderaadt 	case S_IFDIR:
449df930be7Sderaadt 	    return ('/');
450df930be7Sderaadt 
451df930be7Sderaadt 	case S_IFLNK:
452df930be7Sderaadt 	    if (stat(short2str(path), &statb) == 0 &&	/* follow it out */
453df930be7Sderaadt 		S_ISDIR(statb.st_mode))
454df930be7Sderaadt 		return ('>');
455df930be7Sderaadt 	    else
456df930be7Sderaadt 		return ('@');
457df930be7Sderaadt 
458df930be7Sderaadt 	case S_IFSOCK:
459df930be7Sderaadt 	    return ('=');
460df930be7Sderaadt 
461df930be7Sderaadt 	default:
462df930be7Sderaadt 	    if (statb.st_mode & 0111)
463df930be7Sderaadt 		return ('*');
464df930be7Sderaadt 	}
465df930be7Sderaadt     }
466df930be7Sderaadt     return (' ');
467df930be7Sderaadt }
468df930be7Sderaadt 
469df930be7Sderaadt /*
470df930be7Sderaadt  * Print sorted down columns
471df930be7Sderaadt  */
472df930be7Sderaadt static void
print_by_column(Char * dir,Char * items[],int count)473e757c91eSderaadt print_by_column(Char *dir, Char *items[], int count)
474df930be7Sderaadt {
4750134f510Santon     struct winsize win;
476e757c91eSderaadt     int i, rows, r, c, maxwidth = 0, columns;
477df930be7Sderaadt 
4783aaa63ebSderaadt     if (ioctl(SHOUT, TIOCGWINSZ, (ioctl_t) & win) == -1 || win.ws_col == 0)
479df930be7Sderaadt 	win.ws_col = 80;
480df930be7Sderaadt     for (i = 0; i < count; i++)
481df930be7Sderaadt 	maxwidth = maxwidth > (r = Strlen(items[i])) ? maxwidth : r;
482df930be7Sderaadt     maxwidth += 2;		/* for the file tag and space */
483df930be7Sderaadt     columns = win.ws_col / maxwidth;
484df930be7Sderaadt     if (columns == 0)
485df930be7Sderaadt 	columns = 1;
486df930be7Sderaadt     rows = (count + (columns - 1)) / columns;
487df930be7Sderaadt     for (r = 0; r < rows; r++) {
488df930be7Sderaadt 	for (c = 0; c < columns; c++) {
489df930be7Sderaadt 	    i = c * rows + r;
490df930be7Sderaadt 	    if (i < count) {
491e757c91eSderaadt 		int w;
492df930be7Sderaadt 
493df930be7Sderaadt 		(void) fprintf(cshout, "%s", vis_str(items[i]));
494df930be7Sderaadt 		(void) fputc(dir ? filetype(dir, items[i]) : ' ', cshout);
495df930be7Sderaadt 		if (c < columns - 1) {	/* last column? */
496df930be7Sderaadt 		    w = Strlen(items[i]) + 1;
497df930be7Sderaadt 		    for (; w < maxwidth; w++)
498df930be7Sderaadt 			(void) fputc(' ', cshout);
499df930be7Sderaadt 		}
500df930be7Sderaadt 	    }
501df930be7Sderaadt 	}
502df930be7Sderaadt 	(void) fputc('\r', cshout);
503df930be7Sderaadt 	(void) fputc('\n', cshout);
504df930be7Sderaadt     }
505df930be7Sderaadt }
506df930be7Sderaadt 
507df930be7Sderaadt /*
508df930be7Sderaadt  * Expand file name with possible tilde usage
509df930be7Sderaadt  *	~person/mumble
510df930be7Sderaadt  * expands to
511df930be7Sderaadt  *	home_directory_of_person/mumble
512df930be7Sderaadt  */
513df930be7Sderaadt static Char *
tilde(Char * new,Char * old)514e757c91eSderaadt tilde(Char *new, Char *old)
515df930be7Sderaadt {
516e757c91eSderaadt     Char *o, *p;
517e757c91eSderaadt     struct passwd *pw;
518df930be7Sderaadt     static Char person[40];
519df930be7Sderaadt 
5206a01f4acSderaadt     if (old[0] != '~') {
521b9fc9a72Sderaadt 	Strlcpy(new, old, PATH_MAX);
5226a01f4acSderaadt 	return new;
5236a01f4acSderaadt     }
524df930be7Sderaadt 
525df930be7Sderaadt     for (p = person, o = &old[1]; *o && *o != '/'; *p++ = *o++)
526df930be7Sderaadt 	continue;
527df930be7Sderaadt     *p = '\0';
528df930be7Sderaadt     if (person[0] == '\0')
529b9fc9a72Sderaadt 	(void) Strlcpy(new, value(STRhome), PATH_MAX);
530df930be7Sderaadt     else {
531df930be7Sderaadt 	pw = getpwnam(short2str(person));
532df930be7Sderaadt 	if (pw == NULL)
533df930be7Sderaadt 	    return (NULL);
534b9fc9a72Sderaadt 	(void) Strlcpy(new, str2short(pw->pw_dir), PATH_MAX);
535df930be7Sderaadt     }
536b9fc9a72Sderaadt     (void) Strlcat(new, o, PATH_MAX);
537df930be7Sderaadt     return (new);
538df930be7Sderaadt }
539df930be7Sderaadt 
540df930be7Sderaadt /*
541df930be7Sderaadt  * Parse full path in file into 2 parts: directory and file names
542df930be7Sderaadt  * Should leave final slash (/) at end of dir.
543df930be7Sderaadt  */
544df930be7Sderaadt static void
extract_dir_and_name(Char * path,Char * dir,Char * name)545e757c91eSderaadt extract_dir_and_name(Char *path, Char *dir, Char *name)
546df930be7Sderaadt {
547e757c91eSderaadt     Char *p;
548df930be7Sderaadt 
549df930be7Sderaadt     p = Strrchr(path, '/');
550df930be7Sderaadt     if (p == NULL) {
551df930be7Sderaadt 	copyn(name, path, MAXNAMLEN);
552df930be7Sderaadt 	dir[0] = '\0';
553df930be7Sderaadt     }
554df930be7Sderaadt     else {
555df930be7Sderaadt 	copyn(name, ++p, MAXNAMLEN);
556df930be7Sderaadt 	copyn(dir, path, p - path);
557df930be7Sderaadt     }
558df930be7Sderaadt }
559df930be7Sderaadt 
560df930be7Sderaadt static Char *
getentry(DIR * dir_fd,int looking_for_lognames)561e757c91eSderaadt getentry(DIR *dir_fd, int looking_for_lognames)
562df930be7Sderaadt {
563e757c91eSderaadt     struct passwd *pw;
564e757c91eSderaadt     struct dirent *dirp;
565df930be7Sderaadt 
566df930be7Sderaadt     if (looking_for_lognames) {
567df930be7Sderaadt 	if ((pw = getpwent()) == NULL)
568df930be7Sderaadt 	    return (NULL);
569df930be7Sderaadt 	return (str2short(pw->pw_name));
570df930be7Sderaadt     }
571df930be7Sderaadt     if ((dirp = readdir(dir_fd)) != NULL)
572df930be7Sderaadt 	return (str2short(dirp->d_name));
573df930be7Sderaadt     return (NULL);
574df930be7Sderaadt }
575df930be7Sderaadt 
576df930be7Sderaadt static void
free_items(Char ** items,int numitems)577e757c91eSderaadt free_items(Char **items, int numitems)
578df930be7Sderaadt {
579b1f6b197Smillert     int i;
580df930be7Sderaadt 
581b1f6b197Smillert     for (i = 0; i < numitems; i++)
582acdb3202Smestre 	free(items[i]);
583acdb3202Smestre     free(items);
584df930be7Sderaadt }
585df930be7Sderaadt 
586df930be7Sderaadt #define FREE_ITEMS(items) { \
587df930be7Sderaadt 	sigset_t sigset, osigset;\
588df930be7Sderaadt \
589df930be7Sderaadt 	sigemptyset(&sigset);\
590df930be7Sderaadt 	sigaddset(&sigset, SIGINT);\
591df930be7Sderaadt 	sigprocmask(SIG_BLOCK, &sigset, &osigset);\
592b1f6b197Smillert 	free_items(items, numitems);\
593df930be7Sderaadt 	sigprocmask(SIG_SETMASK, &osigset, NULL);\
594df930be7Sderaadt }
595df930be7Sderaadt 
596df930be7Sderaadt /*
597df930be7Sderaadt  * Perform a RECOGNIZE or LIST command on string "word".
598df930be7Sderaadt  */
599df930be7Sderaadt static int
tsearch(Char * word,COMMAND command,int max_word_length)600e757c91eSderaadt tsearch(Char *word, COMMAND command, int max_word_length)
601df930be7Sderaadt {
602e757c91eSderaadt     DIR *dir_fd;
603e757c91eSderaadt     int numitems = 0, ignoring = TRUE, nignored = 0;
604e757c91eSderaadt     int name_length, looking_for_lognames;
605b9fc9a72Sderaadt     Char    tilded_dir[PATH_MAX], dir[PATH_MAX];
606df930be7Sderaadt     Char    name[MAXNAMLEN + 1], extended_name[MAXNAMLEN + 1];
607df930be7Sderaadt     Char   *entry;
6089a5c8861Smillert     Char   **items = NULL;
6099a5c8861Smillert     size_t  maxitems = 0;
610df930be7Sderaadt 
611df930be7Sderaadt     looking_for_lognames = (*word == '~') && (Strchr(word, '/') == NULL);
612df930be7Sderaadt     if (looking_for_lognames) {
613df930be7Sderaadt 	(void) setpwent();
614df930be7Sderaadt 	copyn(name, &word[1], MAXNAMLEN);	/* name sans ~ */
615df930be7Sderaadt 	dir_fd = NULL;
616df930be7Sderaadt     }
617df930be7Sderaadt     else {
618df930be7Sderaadt 	extract_dir_and_name(word, dir, name);
619df930be7Sderaadt 	if (tilde(tilded_dir, dir) == 0)
620df930be7Sderaadt 	    return (0);
621df930be7Sderaadt 	dir_fd = opendir(*tilded_dir ? short2str(tilded_dir) : ".");
622df930be7Sderaadt 	if (dir_fd == NULL)
623df930be7Sderaadt 	    return (0);
624df930be7Sderaadt     }
625df930be7Sderaadt 
626df930be7Sderaadt again:				/* search for matches */
627df930be7Sderaadt     name_length = Strlen(name);
628df930be7Sderaadt     for (numitems = 0; (entry = getentry(dir_fd, looking_for_lognames)) != NULL;) {
629df930be7Sderaadt 	if (!is_prefix(name, entry))
630df930be7Sderaadt 	    continue;
631df930be7Sderaadt 	/* Don't match . files on null prefix match */
632df930be7Sderaadt 	if (name_length == 0 && entry[0] == '.' &&
633df930be7Sderaadt 	    !looking_for_lognames)
634df930be7Sderaadt 	    continue;
635df930be7Sderaadt 	if (command == LIST) {
6369a5c8861Smillert 	    if (numitems >= maxitems) {
6379a5c8861Smillert 		maxitems += 1024;
638a8627d2cSderaadt 		items = xreallocarray(items, maxitems, sizeof(*items));
6399a5c8861Smillert 	    }
640a8627d2cSderaadt 	    items[numitems] = xreallocarray(NULL, (Strlen(entry) + 1), sizeof(Char));
641df930be7Sderaadt 	    copyn(items[numitems], entry, MAXNAMLEN);
642df930be7Sderaadt 	    numitems++;
643df930be7Sderaadt 	}
644df930be7Sderaadt 	else {			/* RECOGNIZE command */
645df930be7Sderaadt 	    if (ignoring && ignored(entry))
646df930be7Sderaadt 		nignored++;
647df930be7Sderaadt 	    else if (recognize(extended_name,
648df930be7Sderaadt 			       entry, name_length, ++numitems))
649df930be7Sderaadt 		break;
650df930be7Sderaadt 	}
651df930be7Sderaadt     }
652df930be7Sderaadt     if (ignoring && numitems == 0 && nignored > 0) {
653df930be7Sderaadt 	ignoring = FALSE;
654df930be7Sderaadt 	nignored = 0;
655df930be7Sderaadt 	if (looking_for_lognames)
656df930be7Sderaadt 	    (void) setpwent();
657df930be7Sderaadt 	else
658df930be7Sderaadt 	    rewinddir(dir_fd);
659df930be7Sderaadt 	goto again;
660df930be7Sderaadt     }
661df930be7Sderaadt 
662df930be7Sderaadt     if (looking_for_lognames)
663df930be7Sderaadt 	(void) endpwent();
664df930be7Sderaadt     else
665df930be7Sderaadt 	(void) closedir(dir_fd);
666df930be7Sderaadt     if (numitems == 0)
667df930be7Sderaadt 	return (0);
668df930be7Sderaadt     if (command == RECOGNIZE) {
669df930be7Sderaadt 	if (looking_for_lognames)
670df930be7Sderaadt 	    copyn(word, STRtilde, 1);
671df930be7Sderaadt 	else
672df930be7Sderaadt 	    /* put back dir part */
673df930be7Sderaadt 	    copyn(word, dir, max_word_length);
674df930be7Sderaadt 	/* add extended name */
675df930be7Sderaadt 	catn(word, extended_name, max_word_length);
676df930be7Sderaadt 	return (numitems);
677df930be7Sderaadt     }
678df930be7Sderaadt     else {			/* LIST */
679246157c1Santon 	qsort(items, numitems, sizeof(*items), sortscmp);
680df930be7Sderaadt 	print_by_column(looking_for_lognames ? NULL : tilded_dir,
681df930be7Sderaadt 			items, numitems);
682df930be7Sderaadt 	if (items != NULL)
683df930be7Sderaadt 	    FREE_ITEMS(items);
684df930be7Sderaadt     }
685df930be7Sderaadt     return (0);
686df930be7Sderaadt }
687df930be7Sderaadt 
688df930be7Sderaadt /*
689df930be7Sderaadt  * Object: extend what user typed up to an ambiguity.
690df930be7Sderaadt  * Algorithm:
691df930be7Sderaadt  * On first match, copy full entry (assume it'll be the only match)
692df930be7Sderaadt  * On subsequent matches, shorten extended_name to the first
693df930be7Sderaadt  * Character mismatch between extended_name and entry.
694df930be7Sderaadt  * If we shorten it back to the prefix length, stop searching.
695df930be7Sderaadt  */
696df930be7Sderaadt static int
recognize(Char * extended_name,Char * entry,int name_length,int numitems)697e757c91eSderaadt recognize(Char *extended_name, Char *entry, int name_length, int numitems)
698df930be7Sderaadt {
699df930be7Sderaadt     if (numitems == 1)		/* 1st match */
700df930be7Sderaadt 	copyn(extended_name, entry, MAXNAMLEN);
701df930be7Sderaadt     else {			/* 2nd & subsequent matches */
702e757c91eSderaadt 	Char *x, *ent;
703e757c91eSderaadt 	int len = 0;
704df930be7Sderaadt 
705df930be7Sderaadt 	x = extended_name;
706df930be7Sderaadt 	for (ent = entry; *x && *x == *ent++; x++, len++)
707df930be7Sderaadt 	    continue;
708df930be7Sderaadt 	*x = '\0';		/* Shorten at 1st Char diff */
709df930be7Sderaadt 	if (len == name_length)	/* Ambiguous to prefix? */
710df930be7Sderaadt 	    return (-1);	/* So stop now and save time */
711df930be7Sderaadt     }
712df930be7Sderaadt     return (0);
713df930be7Sderaadt }
714df930be7Sderaadt 
715df930be7Sderaadt /*
716df930be7Sderaadt  * Return true if check matches initial Chars in template.
717df930be7Sderaadt  * This differs from PWB imatch in that if check is null
718df930be7Sderaadt  * it matches anything.
719df930be7Sderaadt  */
720df930be7Sderaadt static int
is_prefix(Char * check,Char * template)721e757c91eSderaadt is_prefix(Char *check, Char *template)
722df930be7Sderaadt {
723df930be7Sderaadt     do
724df930be7Sderaadt 	if (*check == 0)
725df930be7Sderaadt 	    return (TRUE);
726df930be7Sderaadt     while (*check++ == *template++);
727df930be7Sderaadt     return (FALSE);
728df930be7Sderaadt }
729df930be7Sderaadt 
730df930be7Sderaadt /*
731df930be7Sderaadt  *  Return true if the Chars in template appear at the
732b11b5babSnayden  *  end of check, I.e., are its suffix.
733df930be7Sderaadt  */
734df930be7Sderaadt static int
is_suffix(Char * check,Char * template)735e757c91eSderaadt is_suffix(Char *check, Char *template)
736df930be7Sderaadt {
737e757c91eSderaadt     Char *c, *t;
738df930be7Sderaadt 
739df930be7Sderaadt     for (c = check; *c++;)
740df930be7Sderaadt 	continue;
741df930be7Sderaadt     for (t = template; *t++;)
742df930be7Sderaadt 	continue;
743df930be7Sderaadt     for (;;) {
744df930be7Sderaadt 	if (t == template)
745df930be7Sderaadt 	    return 1;
746df930be7Sderaadt 	if (c == check || *--t != *--c)
747df930be7Sderaadt 	    return 0;
748df930be7Sderaadt     }
749df930be7Sderaadt }
750df930be7Sderaadt 
751df930be7Sderaadt int
tenex(Char * inputline,int inputline_size)752e757c91eSderaadt tenex(Char *inputline, int inputline_size)
753df930be7Sderaadt {
754c0d918cdSanton 	static struct {
755c0d918cdSanton 		int	(*fn)(struct cmdline *, int);
756c0d918cdSanton 		int	idx;
757c0d918cdSanton 	}			 keys[] = {
758c0d918cdSanton 		{ cl_abort,	VINTR },
759c0d918cdSanton 		{ cl_erasec,	VERASE },
760c0d918cdSanton 		{ cl_erasew,	VWERASE },
761c0d918cdSanton 		{ cl_kill,	VKILL },
762c0d918cdSanton 		{ cl_list,	VEOF },
763c0d918cdSanton 		{ cl_literal,	VLNEXT },
764c0d918cdSanton 		{ cl_recognize,	VEOL },
765c0d918cdSanton 		{ cl_reprint,	VREPRINT },
766c0d918cdSanton 		{ cl_status,	VSTATUS },
767c0d918cdSanton 		{ cl_insert,	-1 }
768c0d918cdSanton 	};
769c51b6a6dSderaadt 	unsigned char		 buf[BUFSIZ];
770c0d918cdSanton 	const struct termios	*tio;
771c0d918cdSanton 	struct cmdline		 cl;
772c0d918cdSanton 	size_t			 i;
773c0d918cdSanton 	int			 c, ret;
774df930be7Sderaadt 
775c0d918cdSanton 	memset(&cl, 0, sizeof(cl));
776c0d918cdSanton 	cl.fdin = SHIN;
777c0d918cdSanton 	cl.fdout = SHOUT;
778*157d47b0Sderaadt 	cl.istty = isatty(SHIN);
779*157d47b0Sderaadt 
780*157d47b0Sderaadt 	if (cl.istty)
781*157d47b0Sderaadt 		tio = setup_tty(1);
782*157d47b0Sderaadt 
783c0d918cdSanton 	cl.buf = buf;
784c0d918cdSanton 	cl.size = sizeof(buf);
7857d12c80aSanton 	if (inputline_size < cl.size)
7867d12c80aSanton 		cl.size = inputline_size;
787*157d47b0Sderaadt 	if (cl.istty && tio->c_lflag & ALTWERASE)
788c0d918cdSanton 		cl.flags |= CL_ALTWERASE;
7894a8db713Santon 	if (needprompt) {
7904a8db713Santon 		needprompt = 0;
791fc253c1dSanton 		cl.flags |= CL_PROMPT;
7924a8db713Santon 		cl_flush(&cl);
7934a8db713Santon 	}
794df930be7Sderaadt 
795c0d918cdSanton 	for (;;) {
796c0d918cdSanton 		if ((c = cl_getc(&cl)) == 0)
797df930be7Sderaadt 			break;
798c0d918cdSanton 
799c0d918cdSanton 		for (i = 0; keys[i].idx >= 0; i++)
800*157d47b0Sderaadt 			if (cl.istty && CCEQ(tio->c_cc[keys[i].idx], c))
801df930be7Sderaadt 				break;
802c0d918cdSanton 		ret = keys[i].fn(&cl, c);
803c0d918cdSanton 		cl_flush(&cl);
804c0d918cdSanton 		if (ret)
805c0d918cdSanton 			break;
806df930be7Sderaadt 	}
807df930be7Sderaadt 
808*157d47b0Sderaadt 	if (cl.istty)
809c0d918cdSanton 		setup_tty(0);
810c0d918cdSanton 
811c0d918cdSanton 	for (i = 0; i < cl.len; i++)
812c0d918cdSanton 		inputline[i] = cl.buf[i];
8134a8db713Santon 	/*
8144a8db713Santon 	 * NUL-terminating the buffer implies that it contains a complete
8154a8db713Santon 	 * command ready to be executed. Therefore, don't terminate if the
8164a8db713Santon 	 * buffer is full since more characters must be read in order to form a
8174a8db713Santon 	 * complete command.
8184a8db713Santon 	 */
8196505547eSanton 	if (i < cl.size)
820c0d918cdSanton 		inputline[i] = '\0';
821c0d918cdSanton 
822c0d918cdSanton 	return cl.len;
823df930be7Sderaadt }
824df930be7Sderaadt 
825df930be7Sderaadt static int
ignored(Char * entry)826e757c91eSderaadt ignored(Char *entry)
827df930be7Sderaadt {
828df930be7Sderaadt     struct varent *vp;
829e757c91eSderaadt     Char **cp;
830df930be7Sderaadt 
831df930be7Sderaadt     if ((vp = adrof(STRfignore)) == NULL || (cp = vp->vec) == NULL)
832df930be7Sderaadt 	return (FALSE);
833df930be7Sderaadt     for (; *cp != NULL; cp++)
834df930be7Sderaadt 	if (is_suffix(entry, *cp))
835df930be7Sderaadt 	    return (TRUE);
836df930be7Sderaadt     return (FALSE);
837df930be7Sderaadt }
838