xref: /freebsd/contrib/nvi/vi/getc.c (revision b8ba871b)
1b8ba871bSPeter Wemm /*-
2b8ba871bSPeter Wemm  * Copyright (c) 1992, 1993, 1994
3b8ba871bSPeter Wemm  *	The Regents of the University of California.  All rights reserved.
4b8ba871bSPeter Wemm  * Copyright (c) 1992, 1993, 1994, 1995, 1996
5b8ba871bSPeter Wemm  *	Keith Bostic.  All rights reserved.
6b8ba871bSPeter Wemm  *
7b8ba871bSPeter Wemm  * See the LICENSE file for redistribution information.
8b8ba871bSPeter Wemm  */
9b8ba871bSPeter Wemm 
10b8ba871bSPeter Wemm #include "config.h"
11b8ba871bSPeter Wemm 
12b8ba871bSPeter Wemm #include <sys/types.h>
13b8ba871bSPeter Wemm #include <sys/queue.h>
14b8ba871bSPeter Wemm #include <sys/time.h>
15b8ba871bSPeter Wemm 
16b8ba871bSPeter Wemm #include <bitstring.h>
17b8ba871bSPeter Wemm #include <ctype.h>
18b8ba871bSPeter Wemm #include <limits.h>
19b8ba871bSPeter Wemm #include <stdio.h>
20b8ba871bSPeter Wemm #include <stdlib.h>
21b8ba871bSPeter Wemm 
22b8ba871bSPeter Wemm #include "../common/common.h"
23b8ba871bSPeter Wemm #include "vi.h"
24b8ba871bSPeter Wemm 
25b8ba871bSPeter Wemm /*
26b8ba871bSPeter Wemm  * Character stream routines --
27b8ba871bSPeter Wemm  *	These routines return the file a character at a time.  There are two
28b8ba871bSPeter Wemm  *	special cases.  First, the end of a line, end of a file, start of a
29b8ba871bSPeter Wemm  *	file and empty lines are returned as special cases, and no character
30b8ba871bSPeter Wemm  *	is returned.  Second, empty lines include lines that have only white
31b8ba871bSPeter Wemm  *	space in them, because the vi search functions don't care about white
32b8ba871bSPeter Wemm  *	space, and this makes it easier for them to be consistent.
33b8ba871bSPeter Wemm  */
34b8ba871bSPeter Wemm 
35b8ba871bSPeter Wemm /*
36b8ba871bSPeter Wemm  * cs_init --
37b8ba871bSPeter Wemm  *	Initialize character stream routines.
38b8ba871bSPeter Wemm  *
39b8ba871bSPeter Wemm  * PUBLIC: int cs_init(SCR *, VCS *);
40b8ba871bSPeter Wemm  */
41b8ba871bSPeter Wemm int
cs_init(SCR * sp,VCS * csp)42b8ba871bSPeter Wemm cs_init(SCR *sp, VCS *csp)
43b8ba871bSPeter Wemm {
44b8ba871bSPeter Wemm 	int isempty;
45b8ba871bSPeter Wemm 
46b8ba871bSPeter Wemm 	if (db_eget(sp, csp->cs_lno, &csp->cs_bp, &csp->cs_len, &isempty)) {
47b8ba871bSPeter Wemm 		if (isempty)
48b8ba871bSPeter Wemm 			msgq(sp, M_BERR, "177|Empty file");
49b8ba871bSPeter Wemm 		return (1);
50b8ba871bSPeter Wemm 	}
51b8ba871bSPeter Wemm 	if (csp->cs_len == 0 || v_isempty(csp->cs_bp, csp->cs_len)) {
52b8ba871bSPeter Wemm 		csp->cs_cno = 0;
53b8ba871bSPeter Wemm 		csp->cs_flags = CS_EMP;
54b8ba871bSPeter Wemm 	} else {
55b8ba871bSPeter Wemm 		csp->cs_flags = 0;
56b8ba871bSPeter Wemm 		csp->cs_ch = csp->cs_bp[csp->cs_cno];
57b8ba871bSPeter Wemm 	}
58b8ba871bSPeter Wemm 	return (0);
59b8ba871bSPeter Wemm }
60b8ba871bSPeter Wemm 
61b8ba871bSPeter Wemm /*
62b8ba871bSPeter Wemm  * cs_next --
63b8ba871bSPeter Wemm  *	Retrieve the next character.
64b8ba871bSPeter Wemm  *
65b8ba871bSPeter Wemm  * PUBLIC: int cs_next(SCR *, VCS *);
66b8ba871bSPeter Wemm  */
67b8ba871bSPeter Wemm int
cs_next(SCR * sp,VCS * csp)68b8ba871bSPeter Wemm cs_next(SCR *sp, VCS *csp)
69b8ba871bSPeter Wemm {
70b8ba871bSPeter Wemm 	CHAR_T *p;
71b8ba871bSPeter Wemm 
72b8ba871bSPeter Wemm 	switch (csp->cs_flags) {
73b8ba871bSPeter Wemm 	case CS_EMP:				/* EMP; get next line. */
74b8ba871bSPeter Wemm 	case CS_EOL:				/* EOL; get next line. */
75b8ba871bSPeter Wemm 		if (db_get(sp, ++csp->cs_lno, 0, &p, &csp->cs_len)) {
76b8ba871bSPeter Wemm 			--csp->cs_lno;
77b8ba871bSPeter Wemm 			csp->cs_flags = CS_EOF;
78b8ba871bSPeter Wemm 		} else {
79b8ba871bSPeter Wemm 			csp->cs_bp = p;
80b8ba871bSPeter Wemm 			if (csp->cs_len == 0 ||
81b8ba871bSPeter Wemm 			    v_isempty(csp->cs_bp, csp->cs_len)) {
82b8ba871bSPeter Wemm 				csp->cs_cno = 0;
83b8ba871bSPeter Wemm 				csp->cs_flags = CS_EMP;
84b8ba871bSPeter Wemm 			} else {
85b8ba871bSPeter Wemm 				csp->cs_flags = 0;
86b8ba871bSPeter Wemm 				csp->cs_ch = csp->cs_bp[csp->cs_cno = 0];
87b8ba871bSPeter Wemm 			}
88b8ba871bSPeter Wemm 		}
89b8ba871bSPeter Wemm 		break;
90b8ba871bSPeter Wemm 	case 0:
91b8ba871bSPeter Wemm 		if (csp->cs_cno == csp->cs_len - 1)
92b8ba871bSPeter Wemm 			csp->cs_flags = CS_EOL;
93b8ba871bSPeter Wemm 		else
94b8ba871bSPeter Wemm 			csp->cs_ch = csp->cs_bp[++csp->cs_cno];
95b8ba871bSPeter Wemm 		break;
96b8ba871bSPeter Wemm 	case CS_EOF:				/* EOF. */
97b8ba871bSPeter Wemm 		break;
98b8ba871bSPeter Wemm 	default:
99b8ba871bSPeter Wemm 		abort();
100b8ba871bSPeter Wemm 		/* NOTREACHED */
101b8ba871bSPeter Wemm 	}
102b8ba871bSPeter Wemm 	return (0);
103b8ba871bSPeter Wemm }
104b8ba871bSPeter Wemm 
105b8ba871bSPeter Wemm /*
106b8ba871bSPeter Wemm  * cs_fspace --
107b8ba871bSPeter Wemm  *	If on a space, eat forward until something other than a
108b8ba871bSPeter Wemm  *	whitespace character.
109b8ba871bSPeter Wemm  *
110b8ba871bSPeter Wemm  * XXX
111b8ba871bSPeter Wemm  * Semantics of checking the current character were coded for the fword()
112b8ba871bSPeter Wemm  * function -- once the other word routines are converted, they may have
113b8ba871bSPeter Wemm  * to change.
114b8ba871bSPeter Wemm  *
115b8ba871bSPeter Wemm  * PUBLIC: int cs_fspace(SCR *, VCS *);
116b8ba871bSPeter Wemm  */
117b8ba871bSPeter Wemm int
cs_fspace(SCR * sp,VCS * csp)118b8ba871bSPeter Wemm cs_fspace(SCR *sp, VCS *csp)
119b8ba871bSPeter Wemm {
120b8ba871bSPeter Wemm 	if (csp->cs_flags != 0 || !ISBLANK(csp->cs_ch))
121b8ba871bSPeter Wemm 		return (0);
122b8ba871bSPeter Wemm 	for (;;) {
123b8ba871bSPeter Wemm 		if (cs_next(sp, csp))
124b8ba871bSPeter Wemm 			return (1);
125b8ba871bSPeter Wemm 		if (csp->cs_flags != 0 || !ISBLANK(csp->cs_ch))
126b8ba871bSPeter Wemm 			break;
127b8ba871bSPeter Wemm 	}
128b8ba871bSPeter Wemm 	return (0);
129b8ba871bSPeter Wemm }
130b8ba871bSPeter Wemm 
131b8ba871bSPeter Wemm /*
132b8ba871bSPeter Wemm  * cs_fblank --
133b8ba871bSPeter Wemm  *	Eat forward to the next non-whitespace character.
134b8ba871bSPeter Wemm  *
135b8ba871bSPeter Wemm  * PUBLIC: int cs_fblank(SCR *, VCS *);
136b8ba871bSPeter Wemm  */
137b8ba871bSPeter Wemm int
cs_fblank(SCR * sp,VCS * csp)138b8ba871bSPeter Wemm cs_fblank(SCR *sp, VCS *csp)
139b8ba871bSPeter Wemm {
140b8ba871bSPeter Wemm 	for (;;) {
141b8ba871bSPeter Wemm 		if (cs_next(sp, csp))
142b8ba871bSPeter Wemm 			return (1);
143b8ba871bSPeter Wemm 		if (csp->cs_flags == CS_EOL || csp->cs_flags == CS_EMP ||
144b8ba871bSPeter Wemm 		    (csp->cs_flags == 0 && ISBLANK(csp->cs_ch)))
145b8ba871bSPeter Wemm 			continue;
146b8ba871bSPeter Wemm 		break;
147b8ba871bSPeter Wemm 	}
148b8ba871bSPeter Wemm 	return (0);
149b8ba871bSPeter Wemm }
150b8ba871bSPeter Wemm 
151b8ba871bSPeter Wemm /*
152b8ba871bSPeter Wemm  * cs_prev --
153b8ba871bSPeter Wemm  *	Retrieve the previous character.
154b8ba871bSPeter Wemm  *
155b8ba871bSPeter Wemm  * PUBLIC: int cs_prev(SCR *, VCS *);
156b8ba871bSPeter Wemm  */
157b8ba871bSPeter Wemm int
cs_prev(SCR * sp,VCS * csp)158b8ba871bSPeter Wemm cs_prev(SCR *sp, VCS *csp)
159b8ba871bSPeter Wemm {
160b8ba871bSPeter Wemm 	switch (csp->cs_flags) {
161b8ba871bSPeter Wemm 	case CS_EMP:				/* EMP; get previous line. */
162b8ba871bSPeter Wemm 	case CS_EOL:				/* EOL; get previous line. */
163b8ba871bSPeter Wemm 		if (csp->cs_lno == 1) {		/* SOF. */
164b8ba871bSPeter Wemm 			csp->cs_flags = CS_SOF;
165b8ba871bSPeter Wemm 			break;
166b8ba871bSPeter Wemm 		}
167b8ba871bSPeter Wemm 		if (db_get(sp,			/* The line should exist. */
168b8ba871bSPeter Wemm 		    --csp->cs_lno, DBG_FATAL, &csp->cs_bp, &csp->cs_len)) {
169b8ba871bSPeter Wemm 			++csp->cs_lno;
170b8ba871bSPeter Wemm 			return (1);
171b8ba871bSPeter Wemm 		}
172b8ba871bSPeter Wemm 		if (csp->cs_len == 0 || v_isempty(csp->cs_bp, csp->cs_len)) {
173b8ba871bSPeter Wemm 			csp->cs_cno = 0;
174b8ba871bSPeter Wemm 			csp->cs_flags = CS_EMP;
175b8ba871bSPeter Wemm 		} else {
176b8ba871bSPeter Wemm 			csp->cs_flags = 0;
177b8ba871bSPeter Wemm 			csp->cs_cno = csp->cs_len - 1;
178b8ba871bSPeter Wemm 			csp->cs_ch = csp->cs_bp[csp->cs_cno];
179b8ba871bSPeter Wemm 		}
180b8ba871bSPeter Wemm 		break;
181b8ba871bSPeter Wemm 	case CS_EOF:				/* EOF: get previous char. */
182b8ba871bSPeter Wemm 	case 0:
183b8ba871bSPeter Wemm 		if (csp->cs_cno == 0)
184b8ba871bSPeter Wemm 			if (csp->cs_lno == 1)
185b8ba871bSPeter Wemm 				csp->cs_flags = CS_SOF;
186b8ba871bSPeter Wemm 			else
187b8ba871bSPeter Wemm 				csp->cs_flags = CS_EOL;
188b8ba871bSPeter Wemm 		else
189b8ba871bSPeter Wemm 			csp->cs_ch = csp->cs_bp[--csp->cs_cno];
190b8ba871bSPeter Wemm 		break;
191b8ba871bSPeter Wemm 	case CS_SOF:				/* SOF. */
192b8ba871bSPeter Wemm 		break;
193b8ba871bSPeter Wemm 	default:
194b8ba871bSPeter Wemm 		abort();
195b8ba871bSPeter Wemm 		/* NOTREACHED */
196b8ba871bSPeter Wemm 	}
197b8ba871bSPeter Wemm 	return (0);
198b8ba871bSPeter Wemm }
199b8ba871bSPeter Wemm 
200b8ba871bSPeter Wemm /*
201b8ba871bSPeter Wemm  * cs_bblank --
202b8ba871bSPeter Wemm  *	Eat backward to the next non-whitespace character.
203b8ba871bSPeter Wemm  *
204b8ba871bSPeter Wemm  * PUBLIC: int cs_bblank(SCR *, VCS *);
205b8ba871bSPeter Wemm  */
206b8ba871bSPeter Wemm int
cs_bblank(SCR * sp,VCS * csp)207b8ba871bSPeter Wemm cs_bblank(SCR *sp, VCS *csp)
208b8ba871bSPeter Wemm {
209b8ba871bSPeter Wemm 	for (;;) {
210b8ba871bSPeter Wemm 		if (cs_prev(sp, csp))
211b8ba871bSPeter Wemm 			return (1);
212b8ba871bSPeter Wemm 		if (csp->cs_flags == CS_EOL || csp->cs_flags == CS_EMP ||
213b8ba871bSPeter Wemm 		    (csp->cs_flags == 0 && ISBLANK(csp->cs_ch)))
214b8ba871bSPeter Wemm 			continue;
215b8ba871bSPeter Wemm 		break;
216b8ba871bSPeter Wemm 	}
217b8ba871bSPeter Wemm 	return (0);
218b8ba871bSPeter Wemm }
219b8ba871bSPeter Wemm