rename O_DIRECTORY to O_TMP_DIRECTORY to avoid conflict with open option
[nvi.git] / vi / getc.c
blob50b02140cf7fa86188910cef5a3809a05557138e
1 /*-
2 * Copyright (c) 1992, 1993, 1994
3 * The Regents of the University of California. All rights reserved.
4 * Copyright (c) 1992, 1993, 1994, 1995, 1996
5 * Keith Bostic. All rights reserved.
7 * See the LICENSE file for redistribution information.
8 */
10 #include "config.h"
12 #ifndef lint
13 static const char sccsid[] = "$Id: getc.c,v 10.12 2001/06/25 15:19:30 skimo Exp $ (Berkeley) $Date: 2001/06/25 15:19:30 $";
14 #endif /* not lint */
16 #include <sys/types.h>
17 #include <sys/queue.h>
18 #include <sys/time.h>
20 #include <bitstring.h>
21 #include <ctype.h>
22 #include <limits.h>
23 #include <stdio.h>
24 #include <stdlib.h>
26 #include "../common/common.h"
27 #include "vi.h"
30 * Character stream routines --
31 * These routines return the file a character at a time. There are two
32 * special cases. First, the end of a line, end of a file, start of a
33 * file and empty lines are returned as special cases, and no character
34 * is returned. Second, empty lines include lines that have only white
35 * space in them, because the vi search functions don't care about white
36 * space, and this makes it easier for them to be consistent.
40 * cs_init --
41 * Initialize character stream routines.
43 * PUBLIC: int cs_init __P((SCR *, VCS *));
45 int
46 cs_init(SCR *sp, VCS *csp)
48 int isempty;
50 if (db_eget(sp, csp->cs_lno, &csp->cs_bp, &csp->cs_len, &isempty)) {
51 if (isempty)
52 msgq(sp, M_BERR, "177|Empty file");
53 return (1);
55 if (csp->cs_len == 0 || v_isempty(csp->cs_bp, csp->cs_len)) {
56 csp->cs_cno = 0;
57 csp->cs_flags = CS_EMP;
58 } else {
59 csp->cs_flags = 0;
60 csp->cs_ch = csp->cs_bp[csp->cs_cno];
62 return (0);
66 * cs_next --
67 * Retrieve the next character.
69 * PUBLIC: int cs_next __P((SCR *, VCS *));
71 int
72 cs_next(SCR *sp, VCS *csp)
74 CHAR_T *p;
76 switch (csp->cs_flags) {
77 case CS_EMP: /* EMP; get next line. */
78 case CS_EOL: /* EOL; get next line. */
79 if (db_get(sp, ++csp->cs_lno, 0, &p, &csp->cs_len)) {
80 --csp->cs_lno;
81 csp->cs_flags = CS_EOF;
82 } else {
83 csp->cs_bp = p;
84 if (csp->cs_len == 0 ||
85 v_isempty(csp->cs_bp, csp->cs_len)) {
86 csp->cs_cno = 0;
87 csp->cs_flags = CS_EMP;
88 } else {
89 csp->cs_flags = 0;
90 csp->cs_ch = csp->cs_bp[csp->cs_cno = 0];
93 break;
94 case 0:
95 if (csp->cs_cno == csp->cs_len - 1)
96 csp->cs_flags = CS_EOL;
97 else
98 csp->cs_ch = csp->cs_bp[++csp->cs_cno];
99 break;
100 case CS_EOF: /* EOF. */
101 break;
102 default:
103 abort();
104 /* NOTREACHED */
106 return (0);
110 * cs_fspace --
111 * If on a space, eat forward until something other than a
112 * whitespace character.
114 * XXX
115 * Semantics of checking the current character were coded for the fword()
116 * function -- once the other word routines are converted, they may have
117 * to change.
119 * PUBLIC: int cs_fspace __P((SCR *, VCS *));
122 cs_fspace(SCR *sp, VCS *csp)
124 if (csp->cs_flags != 0 || !isblank(csp->cs_ch))
125 return (0);
126 for (;;) {
127 if (cs_next(sp, csp))
128 return (1);
129 if (csp->cs_flags != 0 || !isblank(csp->cs_ch))
130 break;
132 return (0);
136 * cs_fblank --
137 * Eat forward to the next non-whitespace character.
139 * PUBLIC: int cs_fblank __P((SCR *, VCS *));
142 cs_fblank(SCR *sp, VCS *csp)
144 for (;;) {
145 if (cs_next(sp, csp))
146 return (1);
147 if (csp->cs_flags == CS_EOL || csp->cs_flags == CS_EMP ||
148 csp->cs_flags == 0 && isblank(csp->cs_ch))
149 continue;
150 break;
152 return (0);
156 * cs_prev --
157 * Retrieve the previous character.
159 * PUBLIC: int cs_prev __P((SCR *, VCS *));
162 cs_prev(SCR *sp, VCS *csp)
164 switch (csp->cs_flags) {
165 case CS_EMP: /* EMP; get previous line. */
166 case CS_EOL: /* EOL; get previous line. */
167 if (csp->cs_lno == 1) { /* SOF. */
168 csp->cs_flags = CS_SOF;
169 break;
171 if (db_get(sp, /* The line should exist. */
172 --csp->cs_lno, DBG_FATAL, &csp->cs_bp, &csp->cs_len)) {
173 ++csp->cs_lno;
174 return (1);
176 if (csp->cs_len == 0 || v_isempty(csp->cs_bp, csp->cs_len)) {
177 csp->cs_cno = 0;
178 csp->cs_flags = CS_EMP;
179 } else {
180 csp->cs_flags = 0;
181 csp->cs_cno = csp->cs_len - 1;
182 csp->cs_ch = csp->cs_bp[csp->cs_cno];
184 break;
185 case CS_EOF: /* EOF: get previous char. */
186 case 0:
187 if (csp->cs_cno == 0)
188 if (csp->cs_lno == 1)
189 csp->cs_flags = CS_SOF;
190 else
191 csp->cs_flags = CS_EOL;
192 else
193 csp->cs_ch = csp->cs_bp[--csp->cs_cno];
194 break;
195 case CS_SOF: /* SOF. */
196 break;
197 default:
198 abort();
199 /* NOTREACHED */
201 return (0);
205 * cs_bblank --
206 * Eat backward to the next non-whitespace character.
208 * PUBLIC: int cs_bblank __P((SCR *, VCS *));
211 cs_bblank(SCR *sp, VCS *csp)
213 for (;;) {
214 if (cs_prev(sp, csp))
215 return (1);
216 if (csp->cs_flags == CS_EOL || csp->cs_flags == CS_EMP ||
217 csp->cs_flags == 0 && isblank(csp->cs_ch))
218 continue;
219 break;
221 return (0);