Line | Branch | Exec | Source |
---|---|---|---|
1 | #include <stdbool.h> | ||
2 | #include <stdint.h> | ||
3 | #include <string.h> | ||
4 | #include "status.h" | ||
5 | #include "cmdline.h" | ||
6 | #include "search.h" | ||
7 | #include "selection.h" | ||
8 | #include "util/debug.h" | ||
9 | #include "util/macros.h" | ||
10 | #include "util/numtostr.h" | ||
11 | #include "util/utf8.h" | ||
12 | #include "util/xsnprintf.h" | ||
13 | |||
14 | typedef struct { | ||
15 | char *buf; | ||
16 | size_t size; | ||
17 | size_t pos; | ||
18 | size_t separator; | ||
19 | const Window *window; | ||
20 | const GlobalOptions *opts; | ||
21 | const ModeHandler *mode; | ||
22 | } Formatter; | ||
23 | |||
24 | enum { | ||
25 | SHORT_SEPARATOR_SIZE = 1, // For STATUS_SEPARATOR | ||
26 | LONG_SEPARATOR_SIZE = 3, // For STATUS_SEPARATOR_LONG | ||
27 | SEPARATOR_WRITE_SIZE = 4, // See add_separator() | ||
28 | }; | ||
29 | |||
30 | typedef enum { | ||
31 | STATUS_INVALID = 0, | ||
32 | STATUS_ESCAPED_PERCENT, | ||
33 | STATUS_ENCODING, | ||
34 | STATUS_MISC, | ||
35 | STATUS_IS_CRLF, | ||
36 | STATUS_SEPARATOR_LONG, | ||
37 | STATUS_CURSOR_COL_BYTES, | ||
38 | STATUS_TOTAL_ROWS, | ||
39 | STATUS_BOM, | ||
40 | STATUS_FILENAME, | ||
41 | STATUS_INPUT_MODE, | ||
42 | STATUS_MODIFIED, | ||
43 | STATUS_LINE_ENDING, | ||
44 | STATUS_OVERWRITE, | ||
45 | STATUS_SCROLL_POSITION, | ||
46 | STATUS_READONLY, | ||
47 | STATUS_SEPARATOR, | ||
48 | STATUS_FILETYPE, | ||
49 | STATUS_UNICODE, | ||
50 | STATUS_CURSOR_COL, | ||
51 | STATUS_CURSOR_ROW, | ||
52 | } FormatSpecifierType; | ||
53 | |||
54 | 343 | static FormatSpecifierType lookup_format_specifier(unsigned char ch) | |
55 | { | ||
56 |
21/21✓ Branch 0 (2→3) taken 12 times.
✓ Branch 1 (2→4) taken 16 times.
✓ Branch 2 (2→5) taken 6 times.
✓ Branch 3 (2→6) taken 4 times.
✓ Branch 4 (2→7) taken 11 times.
✓ Branch 5 (2→8) taken 4 times.
✓ Branch 6 (2→9) taken 12 times.
✓ Branch 7 (2→10) taken 12 times.
✓ Branch 8 (2→11) taken 2 times.
✓ Branch 9 (2→12) taken 12 times.
✓ Branch 10 (2→13) taken 14 times.
✓ Branch 11 (2→14) taken 12 times.
✓ Branch 12 (2→15) taken 11 times.
✓ Branch 13 (2→16) taken 14 times.
✓ Branch 14 (2→17) taken 55 times.
✓ Branch 15 (2→18) taken 12 times.
✓ Branch 16 (2→19) taken 11 times.
✓ Branch 17 (2→20) taken 3 times.
✓ Branch 18 (2→21) taken 12 times.
✓ Branch 19 (2→22) taken 104 times.
✓ Branch 20 (2→23) taken 4 times.
|
343 | switch (ch) { |
57 | case '%': return STATUS_ESCAPED_PERCENT; | ||
58 | 12 | case 'E': return STATUS_ENCODING; | |
59 | 16 | case 'M': return STATUS_MISC; | |
60 | 6 | case 'N': return STATUS_IS_CRLF; | |
61 | 4 | case 'S': return STATUS_SEPARATOR_LONG; | |
62 | 11 | case 'X': return STATUS_CURSOR_COL_BYTES; | |
63 | 4 | case 'Y': return STATUS_TOTAL_ROWS; | |
64 | 12 | case 'b': return STATUS_BOM; | |
65 | 12 | case 'f': return STATUS_FILENAME; | |
66 | 2 | case 'i': return STATUS_INPUT_MODE; | |
67 | 12 | case 'm': return STATUS_MODIFIED; | |
68 | 14 | case 'n': return STATUS_LINE_ENDING; | |
69 | 12 | case 'o': return STATUS_OVERWRITE; | |
70 | 11 | case 'p': return STATUS_SCROLL_POSITION; | |
71 | 14 | case 'r': return STATUS_READONLY; | |
72 | 55 | case 's': return STATUS_SEPARATOR; | |
73 | 12 | case 't': return STATUS_FILETYPE; | |
74 | 11 | case 'u': return STATUS_UNICODE; | |
75 | 3 | case 'x': return STATUS_CURSOR_COL; | |
76 | 12 | case 'y': return STATUS_CURSOR_ROW; | |
77 | } | ||
78 | 104 | return STATUS_INVALID; | |
79 | } | ||
80 | |||
81 | #define add_status_literal(f, s) add_status_bytes(f, s, STRLEN(s)) | ||
82 | |||
83 | 8 | static void add_ch(Formatter *f, char ch) | |
84 | { | ||
85 | 8 | f->buf[f->pos++] = ch; | |
86 | 8 | } | |
87 | |||
88 | 40 | static void add_separator(Formatter *f) | |
89 | { | ||
90 | 40 | size_t len = MIN(f->separator, f->size - f->pos); | |
91 | 40 | BUG_ON(len > LONG_SEPARATOR_SIZE); | |
92 | 40 | char *dest = f->buf + f->pos; | |
93 | 40 | f->pos += len; | |
94 | 40 | f->separator = 0; | |
95 | |||
96 | // The extra buffer space set aside by sf_format() allows us to write | ||
97 | // 4 bytes unconditionally here, to allow this memset() call to be | ||
98 | // optimized into a simple move instruction | ||
99 | 40 | memset(dest, ' ', SEPARATOR_WRITE_SIZE); | |
100 | 40 | } | |
101 | |||
102 | 7 | static void add_status_str(Formatter *f, const char *str) | |
103 | { | ||
104 | 7 | BUG_ON(!str); | |
105 |
1/2✓ Branch 0 (4→5) taken 7 times.
✗ Branch 1 (4→13) not taken.
|
7 | if (unlikely(!str[0])) { |
106 | return; | ||
107 | } | ||
108 | |||
109 | 7 | add_separator(f); | |
110 | 7 | char *buf = f->buf; | |
111 | 7 | size_t pos = f->pos; | |
112 | |||
113 |
3/4✓ Branch 0 (10→11) taken 43 times.
✗ Branch 1 (10→12) not taken.
✓ Branch 2 (11→7) taken 36 times.
✓ Branch 3 (11→12) taken 7 times.
|
43 | for (size_t i = 0, size = f->size; pos < size && str[i]; ) { |
114 | 36 | pos += u_set_char(buf + pos, u_str_get_char(str, &i)); | |
115 | } | ||
116 | |||
117 | 7 | f->pos = pos; | |
118 | } | ||
119 | |||
120 | 25 | static void add_status_bytes(Formatter *f, const char *str, size_t len) | |
121 | { | ||
122 |
1/2✓ Branch 0 (2→3) taken 25 times.
✗ Branch 1 (2→7) not taken.
|
25 | if (unlikely(len == 0)) { |
123 | return; | ||
124 | } | ||
125 | |||
126 | 25 | add_separator(f); | |
127 |
1/2✓ Branch 0 (4→5) taken 25 times.
✗ Branch 1 (4→7) not taken.
|
25 | if (f->pos >= f->size) { |
128 | return; | ||
129 | } | ||
130 | |||
131 | 25 | size_t avail = f->size - f->pos; | |
132 | 25 | f->pos += copystrn(f->buf + f->pos, str, MIN(len, avail)); | |
133 | } | ||
134 | |||
135 | PRINTF(2) | ||
136 | ✗ | static void add_status_format(Formatter *f, const char *format, ...) | |
137 | { | ||
138 | ✗ | char buf[1024]; | |
139 | ✗ | va_list ap; | |
140 | ✗ | va_start(ap, format); | |
141 | ✗ | size_t len = xvsnprintf(buf, sizeof(buf), format, ap); | |
142 | ✗ | va_end(ap); | |
143 | ✗ | add_status_bytes(f, buf, len); | |
144 | ✗ | } | |
145 | |||
146 | 6 | static void add_status_umax(Formatter *f, uintmax_t x) | |
147 | { | ||
148 | 6 | char buf[DECIMAL_STR_MAX(x)]; | |
149 | 6 | size_t len = buf_umax_to_str(x, buf); | |
150 | 6 | add_status_bytes(f, buf, len); | |
151 | 6 | } | |
152 | |||
153 | 7 | static void add_status_bool(Formatter *f, bool state, const char *on, const char *off) | |
154 | { | ||
155 |
2/2✓ Branch 0 (2→3) taken 3 times.
✓ Branch 1 (2→4) taken 4 times.
|
7 | const char *str = state ? on : off; |
156 | 7 | add_status_bytes(f, str, strlen(str)); | |
157 | 7 | } | |
158 | |||
159 | 1 | static void add_status_unicode(Formatter *f, const BlockIter *cursor) | |
160 | { | ||
161 | 1 | CodePoint u; | |
162 |
1/2✗ Branch 0 (3→4) not taken.
✓ Branch 1 (3→10) taken 1 times.
|
1 | if (unlikely(!block_iter_get_char(cursor, &u))) { |
163 | 1 | return; | |
164 | } | ||
165 | |||
166 | ✗ | if (!u_is_unicode(u)) { | |
167 | ✗ | add_status_literal(f, "Invalid"); | |
168 | ✗ | return; | |
169 | } | ||
170 | |||
171 | ✗ | char str[STRLEN("U+10FFFF") + 1]; | |
172 | ✗ | str[0] = 'U'; | |
173 | ✗ | str[1] = '+'; | |
174 | ✗ | size_t ndigits = buf_umax_to_hex_str(u, str + 2, 4); | |
175 | ✗ | add_status_bytes(f, str, 2 + ndigits); | |
176 | } | ||
177 | |||
178 | 1 | static void add_status_pos(Formatter *f) | |
179 | { | ||
180 | 1 | size_t lines = f->window->view->buffer->nl; | |
181 | 1 | int h = f->window->edit_h; | |
182 | 1 | long pos = f->window->view->vy; | |
183 |
1/2✓ Branch 0 (2→3) taken 1 times.
✗ Branch 1 (2→6) not taken.
|
1 | if (lines <= h) { |
184 |
1/2✗ Branch 0 (3→4) not taken.
✓ Branch 1 (3→5) taken 1 times.
|
1 | if (pos) { |
185 | ✗ | add_status_literal(f, "Bot"); | |
186 | } else { | ||
187 | 1 | add_status_literal(f, "All"); | |
188 | } | ||
189 | ✗ | } else if (pos == 0) { | |
190 | ✗ | add_status_literal(f, "Top"); | |
191 | ✗ | } else if (pos + h - 1 >= lines) { | |
192 | ✗ | add_status_literal(f, "Bot"); | |
193 | } else { | ||
194 | ✗ | unsigned int d = lines - (h - 1); | |
195 | ✗ | unsigned int percent = (pos * 100 + d / 2) / d; | |
196 | ✗ | BUG_ON(percent > 100); | |
197 | ✗ | char buf[4]; | |
198 | ✗ | size_t len = buf_uint_to_str(percent, buf); | |
199 | ✗ | buf[len++] = '%'; | |
200 | ✗ | add_status_bytes(f, buf, len); | |
201 | } | ||
202 | 1 | } | |
203 | |||
204 | 6 | static void add_misc_status(Formatter *f) | |
205 | { | ||
206 | 6 | static const struct { | |
207 | const char str[24]; | ||
208 | size_t len; | ||
209 | } css_strs[] = { | ||
210 | [CSS_FALSE] = {STRN("[case-sensitive = false]")}, | ||
211 | [CSS_TRUE] = {STRN("[case-sensitive = true]")}, | ||
212 | [CSS_AUTO] = {STRN("[case-sensitive = auto]")}, | ||
213 | }; | ||
214 | |||
215 |
2/2✓ Branch 0 (2→3) taken 5 times.
✓ Branch 1 (2→6) taken 1 times.
|
6 | if (f->mode->cmds == &search_mode_commands) { |
216 | 5 | SearchCaseSensitivity css = f->opts->case_sensitive_search; | |
217 | 5 | BUG_ON(css >= ARRAYLEN(css_strs)); | |
218 | 5 | add_status_bytes(f, css_strs[css].str, css_strs[css].len); | |
219 | 11 | return; | |
220 | } | ||
221 | |||
222 | 1 | const View *view = f->window->view; | |
223 |
1/2✗ Branch 0 (6→7) not taken.
✓ Branch 1 (6→18) taken 1 times.
|
1 | if (view->selection == SELECT_NONE) { |
224 | return; | ||
225 | } | ||
226 | |||
227 | ✗ | SelectionInfo si = init_selection(view); | |
228 | ✗ | bool is_lines = (view->selection == SELECT_LINES); | |
229 | ✗ | size_t n = is_lines ? get_nr_selected_lines(&si) : get_nr_selected_chars(&si); | |
230 | ✗ | const char *unit = is_lines ? "line" : "char"; | |
231 | ✗ | const char *plural = unlikely(n == 1) ? "" : "s"; | |
232 | ✗ | add_status_format(f, "[%zu %s%s]", n, unit, plural); | |
233 | } | ||
234 | |||
235 | 59 | static void expand_format_specifier(Formatter *f, FormatSpecifierType type) | |
236 | { | ||
237 | 59 | const View *view = f->window->view; | |
238 | 59 | const Buffer *buffer = view->buffer; | |
239 |
20/21✓ Branch 0 (2→3) taken 3 times.
✓ Branch 1 (2→5) taken 2 times.
✓ Branch 2 (2→7) taken 1 times.
✓ Branch 3 (2→9) taken 2 times.
✓ Branch 4 (2→12) taken 4 times.
✓ Branch 5 (2→16) taken 2 times.
✓ Branch 6 (2→18) taken 2 times.
✓ Branch 7 (2→20) taken 1 times.
✓ Branch 8 (2→22) taken 1 times.
✓ Branch 9 (2→26) taken 1 times.
✓ Branch 10 (2→28) taken 2 times.
✓ Branch 11 (2→30) taken 6 times.
✓ Branch 12 (2→32) taken 4 times.
✓ Branch 13 (2→34) taken 4 times.
✓ Branch 14 (2→36) taken 3 times.
✓ Branch 15 (2→38) taken 3 times.
✓ Branch 16 (2→39) taken 13 times.
✓ Branch 17 (2→40) taken 2 times.
✓ Branch 18 (2→42) taken 1 times.
✓ Branch 19 (2→44) taken 2 times.
✗ Branch 20 (2→47) not taken.
|
59 | switch (type) { |
240 | 3 | case STATUS_BOM: | |
241 |
2/2✓ Branch 0 (3→4) taken 2 times.
✓ Branch 1 (3→49) taken 1 times.
|
3 | if (buffer->bom) { |
242 | 2 | add_status_literal(f, "BOM"); | |
243 | } | ||
244 | return; | ||
245 | 2 | case STATUS_FILENAME: | |
246 | 2 | add_status_str(f, buffer_filename(buffer)); | |
247 | 2 | return; | |
248 | 1 | case STATUS_INPUT_MODE: | |
249 | 1 | add_status_str(f, f->mode->name); | |
250 | 1 | return; | |
251 | 2 | case STATUS_MODIFIED: | |
252 |
1/2✗ Branch 0 (9→10) not taken.
✓ Branch 1 (9→49) taken 2 times.
|
2 | if (buffer_modified(buffer)) { |
253 | ✗ | add_separator(f); | |
254 | ✗ | add_ch(f, '*'); | |
255 | } | ||
256 | return; | ||
257 | 4 | case STATUS_READONLY: | |
258 |
1/2✗ Branch 0 (12→13) not taken.
✓ Branch 1 (12→14) taken 4 times.
|
4 | if (buffer->readonly) { |
259 | ✗ | add_status_literal(f, "RO"); | |
260 |
2/2✓ Branch 0 (14→15) taken 2 times.
✓ Branch 1 (14→49) taken 2 times.
|
4 | } else if (buffer->temporary) { |
261 | 2 | add_status_literal(f, "TMP"); | |
262 | } | ||
263 | return; | ||
264 | 2 | case STATUS_CURSOR_ROW: | |
265 | 2 | add_status_umax(f, view->cy + 1); | |
266 | 2 | return; | |
267 | 2 | case STATUS_TOTAL_ROWS: | |
268 | 2 | add_status_umax(f, buffer->nl); | |
269 | 2 | return; | |
270 | 1 | case STATUS_CURSOR_COL: | |
271 | 1 | add_status_umax(f, view->cx_display + 1); | |
272 | 1 | return; | |
273 | 1 | case STATUS_CURSOR_COL_BYTES: | |
274 | 1 | add_status_umax(f, view->cx_char + 1); | |
275 |
1/2✗ Branch 0 (23→24) not taken.
✓ Branch 1 (23→49) taken 1 times.
|
1 | if (view->cx_display != view->cx_char) { |
276 | ✗ | add_ch(f, '-'); | |
277 | ✗ | add_status_umax(f, view->cx_display + 1); | |
278 | } | ||
279 | return; | ||
280 | 1 | case STATUS_SCROLL_POSITION: | |
281 | 1 | add_status_pos(f); | |
282 | 1 | return; | |
283 | 2 | case STATUS_ENCODING: | |
284 | 2 | add_status_str(f, buffer->encoding); | |
285 | 2 | return; | |
286 | 6 | case STATUS_MISC: | |
287 | 6 | add_misc_status(f); | |
288 | 6 | return; | |
289 | 4 | case STATUS_IS_CRLF: | |
290 |
2/2✓ Branch 0 (32→33) taken 2 times.
✓ Branch 1 (32→49) taken 2 times.
|
4 | if (buffer->crlf_newlines) { |
291 | 2 | add_status_literal(f, "CRLF"); | |
292 | } | ||
293 | return; | ||
294 | 4 | case STATUS_LINE_ENDING: | |
295 | 4 | add_status_bool(f, buffer->crlf_newlines, "CRLF", "LF"); | |
296 | 4 | return; | |
297 | 3 | case STATUS_OVERWRITE: | |
298 | 3 | add_status_bool(f, buffer->options.overwrite, "OVR", "INS"); | |
299 | 3 | return; | |
300 | 3 | case STATUS_SEPARATOR_LONG: | |
301 | 3 | f->separator = LONG_SEPARATOR_SIZE; | |
302 | 3 | return; | |
303 | 13 | case STATUS_SEPARATOR: | |
304 | 13 | f->separator = SHORT_SEPARATOR_SIZE; | |
305 | 13 | return; | |
306 | 2 | case STATUS_FILETYPE: | |
307 | 2 | add_status_str(f, buffer->options.filetype); | |
308 | 2 | return; | |
309 | 1 | case STATUS_UNICODE: | |
310 | 1 | add_status_unicode(f, &view->cursor); | |
311 | 1 | return; | |
312 | 2 | case STATUS_ESCAPED_PERCENT: | |
313 | 2 | add_separator(f); | |
314 | 2 | add_ch(f, '%'); | |
315 | 2 | return; | |
316 | case STATUS_INVALID: | ||
317 | break; | ||
318 | } | ||
319 | |||
320 | − | BUG("should be unreachable, due to validate_statusline_format()"); | |
321 | } | ||
322 | |||
323 | 26 | size_t sf_format ( | |
324 | const Window *window, | ||
325 | const GlobalOptions *opts, | ||
326 | const ModeHandler *mode, | ||
327 | char *buf, // NOLINT(readability-non-const-parameter) | ||
328 | size_t size, | ||
329 | const char *format | ||
330 | ) { | ||
331 | 26 | BUG_ON(size < 16); | |
332 | 26 | Formatter f = { | |
333 | .window = window, | ||
334 | .opts = opts, | ||
335 | .mode = mode, | ||
336 | .buf = buf, | ||
337 | 26 | .size = size - SEPARATOR_WRITE_SIZE - U_SET_CHAR_MAXLEN - 1, | |
338 | }; | ||
339 | |||
340 |
4/4✓ Branch 0 (11→12) taken 90 times.
✓ Branch 1 (11→13) taken 1 times.
✓ Branch 2 (12→5) taken 65 times.
✓ Branch 3 (12→13) taken 25 times.
|
91 | while (f.pos < f.size && *format) { |
341 | 65 | unsigned char ch = *format++; | |
342 |
2/2✓ Branch 0 (5→6) taken 6 times.
✓ Branch 1 (5→9) taken 59 times.
|
65 | if (ch != '%') { |
343 | 6 | add_separator(&f); | |
344 | 6 | add_ch(&f, ch); | |
345 | 6 | continue; | |
346 | } | ||
347 | 59 | FormatSpecifierType type = lookup_format_specifier(*format++); | |
348 | 59 | expand_format_specifier(&f, type); | |
349 | } | ||
350 | |||
351 | 26 | f.buf[f.pos] = '\0'; | |
352 | 26 | return u_str_width(f.buf); | |
353 | } | ||
354 | |||
355 | // Returns the offset of the first invalid format specifier, or 0 if | ||
356 | // the whole format string is valid. It's safe to use 0 to indicate | ||
357 | // "no errors", since it's not possible for there to be an error at | ||
358 | // the very start of the string. | ||
359 | 141 | size_t statusline_format_find_error(const char *str) | |
360 | { | ||
361 |
2/2✓ Branch 0 (7→3) taken 411 times.
✓ Branch 1 (7→8) taken 37 times.
|
448 | for (size_t i = 0; str[i]; ) { |
362 |
2/2✓ Branch 0 (3→4) taken 127 times.
✓ Branch 1 (3→5) taken 284 times.
|
411 | if (str[i++] != '%') { |
363 | 127 | continue; | |
364 | } | ||
365 |
2/2✓ Branch 0 (5→6) taken 180 times.
✓ Branch 1 (5→8) taken 104 times.
|
284 | if (lookup_format_specifier(str[i++]) == STATUS_INVALID) { |
366 | return i - 1; | ||
367 | } | ||
368 | } | ||
369 | return 0; | ||
370 | } | ||
371 |