Rewrite foldline() in fold(1)

After the audit, I had this noted down as a TODO-item, but
considered the function to be tested enough to hold the line
until I came to rewrite it.
Admittedly, I didn't take a closer look at the previous loop
and there probably were some edge-cases which caused trouble, but
so far so good, the new version of this commit should be safe
and considered audited.
This commit is contained in:
FRIGN 2015-03-16 19:26:42 +01:00
parent 64da0d1ebc
commit fbda47b964
1 changed files with 38 additions and 42 deletions

80
fold.c
View File

@ -3,6 +3,7 @@
#include <stdint.h> #include <stdint.h>
#include <stdio.h> #include <stdio.h>
#include <stdlib.h> #include <stdlib.h>
#include <string.h>
#include "util.h" #include "util.h"
@ -11,46 +12,41 @@ static int sflag = 0;
static size_t width = 80; static size_t width = 80;
static void static void
foldline(const char *str) foldline(const char *str) {
{ const char *p, *spacesect = NULL;
size_t i = 0, n = 0, col, j; size_t col, off;
int space;
char c;
do { for (p = str, col = 0; *p && *p != '\n'; p++) {
space = 0; if (!UTF8_POINT(*p) && !bflag)
for (j = i, col = 0; str[j] && col <= width; j++) { continue;
c = str[j]; if (col >= width) {
if (!UTF8_POINT(c) && !bflag) off = (sflag && spacesect) ? spacesect - str : p - str;
continue; if (fwrite(str, 1, off, stdout) != off)
if (sflag && isspace(c)) { eprintf("fwrite <stdout>:");
space = 1;
n = j + 1;
} else if (!space) {
n = j;
}
if (!bflag && iscntrl(c)) {
switch(c) {
case '\b':
col--;
break;
case '\r':
col = 0;
break;
case '\t':
col += (col + 1) % 8;
break;
}
} else {
col++;
}
}
if (fwrite(str + i, 1, n - i, stdout) != n - i)
eprintf("fwrite <stdout>:");
if (str[n])
putchar('\n'); putchar('\n');
} while (str[i = n] && str[i] != '\n'); spacesect = NULL;
col = 0;
p = str += off;
}
if (sflag && isspace(*p))
spacesect = p + 1;
if (!bflag && iscntrl(*p)) {
switch(*p) {
case '\b':
col -= (col > 0);
break;
case '\r':
col = 0;
break;
case '\t':
col += (col + 1) % 8;
break;
}
} else {
col++;
}
}
fputs(str, stdout);
} }
static void static void
@ -69,7 +65,7 @@ fold(FILE *fp, const char *fname)
static void static void
usage(void) usage(void)
{ {
eprintf("usage: %s [-bs] [-w width | -width] [FILE...]\n", argv0); eprintf("usage: %s [-bs] [-w num | -num] [FILE ...]\n", argv0);
} }
int int
@ -102,10 +98,10 @@ main(int argc, char *argv[])
if (!(fp = fopen(*argv, "r"))) { if (!(fp = fopen(*argv, "r"))) {
weprintf("fopen %s:", *argv); weprintf("fopen %s:", *argv);
ret = 1; ret = 1;
continue; } else {
fold(fp, *argv);
fclose(fp);
} }
fold(fp, *argv);
fclose(fp);
} }
} }