Support NUL containing lines in nl(1) and reuse the line-buffer(!)
This commit is contained in:
parent
c25996924b
commit
b58884312c
2
README
2
README
|
@ -56,7 +56,7 @@ The following tools are implemented:
|
||||||
0=*|x mktemp .
|
0=*|x mktemp .
|
||||||
0=*|o mv (-i)
|
0=*|o mv (-i)
|
||||||
0=*|o nice .
|
0=*|o nice .
|
||||||
#*|o nl .
|
0#*|o nl .
|
||||||
0=*|o nohup .
|
0=*|o nohup .
|
||||||
0=*|o od .
|
0=*|o od .
|
||||||
0#* o pathchk .
|
0#* o pathchk .
|
||||||
|
|
47
nl.c
47
nl.c
|
@ -13,20 +13,23 @@ static size_t startnum = 1;
|
||||||
static size_t incr = 1;
|
static size_t incr = 1;
|
||||||
static size_t blines = 1;
|
static size_t blines = 1;
|
||||||
static size_t delimlen = 2;
|
static size_t delimlen = 2;
|
||||||
|
static size_t seplen = 1;
|
||||||
static int width = 6;
|
static int width = 6;
|
||||||
static int pflag = 0;
|
static int pflag = 0;
|
||||||
static char type[] = { 'n', 't', 'n' }; /* footer, body, header */
|
static char type[] = { 'n', 't', 'n' }; /* footer, body, header */
|
||||||
static char *delim = "\\:";
|
static char *delim = "\\:";
|
||||||
static char format[8] = "%*ld%s";
|
static char format[6] = "%*ld";
|
||||||
static char *sep = "\t";
|
static char *sep = "\t";
|
||||||
static regex_t preg[3];
|
static regex_t preg[3];
|
||||||
|
|
||||||
static int
|
static int
|
||||||
getsection(char *buf, int *section)
|
getsection(struct line *l, int *section)
|
||||||
{
|
{
|
||||||
|
size_t i;
|
||||||
int sectionchanged = 0, newsection = *section;
|
int sectionchanged = 0, newsection = *section;
|
||||||
|
|
||||||
for (; !strncmp(buf, delim, delimlen); buf += delimlen) {
|
for (i = 0; (l->len - i) >= delimlen &&
|
||||||
|
!memcmp(l->data + i, delim, delimlen); i += delimlen) {
|
||||||
if (!sectionchanged) {
|
if (!sectionchanged) {
|
||||||
sectionchanged = 1;
|
sectionchanged = 1;
|
||||||
newsection = 0;
|
newsection = 0;
|
||||||
|
@ -35,7 +38,7 @@ getsection(char *buf, int *section)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!buf[0] || buf[0] == '\n')
|
if (!(l->len - i) || l->data[i] == '\n')
|
||||||
*section = newsection;
|
*section = newsection;
|
||||||
else
|
else
|
||||||
sectionchanged = 0;
|
sectionchanged = 0;
|
||||||
|
@ -46,15 +49,18 @@ getsection(char *buf, int *section)
|
||||||
static void
|
static void
|
||||||
nl(const char *fname, FILE *fp)
|
nl(const char *fname, FILE *fp)
|
||||||
{
|
{
|
||||||
size_t number = startnum, size = 0, bl = 1;
|
static struct line line;
|
||||||
|
static size_t size;
|
||||||
|
size_t number = startnum, bl = 1;
|
||||||
|
ssize_t len;
|
||||||
int donumber, oldsection, section = 1;
|
int donumber, oldsection, section = 1;
|
||||||
char *buf = NULL;
|
|
||||||
|
|
||||||
while (getline(&buf, &size, fp) > 0) {
|
while ((len = getline(&line.data, &size, fp)) > 0) {
|
||||||
|
line.len = len;
|
||||||
donumber = 0;
|
donumber = 0;
|
||||||
oldsection = section;
|
oldsection = section;
|
||||||
|
|
||||||
if (getsection(buf, §ion)) {
|
if (getsection(&line, §ion)) {
|
||||||
if ((section >= oldsection) && !pflag)
|
if ((section >= oldsection) && !pflag)
|
||||||
number = startnum;
|
number = startnum;
|
||||||
continue;
|
continue;
|
||||||
|
@ -62,15 +68,15 @@ nl(const char *fname, FILE *fp)
|
||||||
|
|
||||||
switch (type[section]) {
|
switch (type[section]) {
|
||||||
case 't':
|
case 't':
|
||||||
if (buf[0] != '\n')
|
if (line.data[0] != '\n')
|
||||||
donumber = 1;
|
donumber = 1;
|
||||||
break;
|
break;
|
||||||
case 'p':
|
case 'p':
|
||||||
if (!regexec(preg + section, buf, 0, NULL, 0))
|
if (!regexec(preg + section, line.data, 0, NULL, 0))
|
||||||
donumber = 1;
|
donumber = 1;
|
||||||
break;
|
break;
|
||||||
case 'a':
|
case 'a':
|
||||||
if (buf[0] == '\n' && bl < blines) {
|
if (line.data[0] == '\n' && bl < blines) {
|
||||||
++bl;
|
++bl;
|
||||||
} else {
|
} else {
|
||||||
donumber = 1;
|
donumber = 1;
|
||||||
|
@ -79,12 +85,13 @@ nl(const char *fname, FILE *fp)
|
||||||
}
|
}
|
||||||
|
|
||||||
if (donumber) {
|
if (donumber) {
|
||||||
printf(format, width, number, sep);
|
printf(format, width, number);
|
||||||
|
fwrite(sep, 1, seplen, stdout);
|
||||||
number += incr;
|
number += incr;
|
||||||
}
|
}
|
||||||
fputs(buf, stdout);
|
fwrite(line.data, 1, line.len, stdout);
|
||||||
}
|
}
|
||||||
free(buf);
|
free(line.data);
|
||||||
if (ferror(fp))
|
if (ferror(fp))
|
||||||
eprintf("getline %s:", fname);
|
eprintf("getline %s:", fname);
|
||||||
}
|
}
|
||||||
|
@ -112,18 +119,15 @@ int
|
||||||
main(int argc, char *argv[])
|
main(int argc, char *argv[])
|
||||||
{
|
{
|
||||||
FILE *fp = NULL;
|
FILE *fp = NULL;
|
||||||
size_t l, s;
|
size_t s;
|
||||||
int ret = 0;
|
int ret = 0;
|
||||||
char *d, *formattype, *formatblit;
|
char *d, *formattype, *formatblit;
|
||||||
|
|
||||||
ARGBEGIN {
|
ARGBEGIN {
|
||||||
case 'd':
|
case 'd':
|
||||||
d = EARGF(usage());
|
switch (utflen((d = EARGF(usage())))) {
|
||||||
l = utflen(d);
|
|
||||||
|
|
||||||
switch (l) {
|
|
||||||
case 0:
|
case 0:
|
||||||
break;
|
eprintf("empty logical page delimiter\n");
|
||||||
case 1:
|
case 1:
|
||||||
s = strlen(d);
|
s = strlen(d);
|
||||||
delim = emalloc(s + 1 + 1);
|
delim = emalloc(s + 1 + 1);
|
||||||
|
@ -167,13 +171,14 @@ main(int argc, char *argv[])
|
||||||
}
|
}
|
||||||
|
|
||||||
estrlcat(format, formatblit, sizeof(format));
|
estrlcat(format, formatblit, sizeof(format));
|
||||||
estrlcat(format, "*ld%s", sizeof(format));
|
estrlcat(format, "*ld", sizeof(format));
|
||||||
break;
|
break;
|
||||||
case 'p':
|
case 'p':
|
||||||
pflag = 1;
|
pflag = 1;
|
||||||
break;
|
break;
|
||||||
case 's':
|
case 's':
|
||||||
sep = EARGF(usage());
|
sep = EARGF(usage());
|
||||||
|
seplen = unescape(sep);
|
||||||
break;
|
break;
|
||||||
case 'v':
|
case 'v':
|
||||||
startnum = estrtonum(EARGF(usage()), 0, MIN(LLONG_MAX, SIZE_MAX));
|
startnum = estrtonum(EARGF(usage()), 0, MIN(LLONG_MAX, SIZE_MAX));
|
||||||
|
|
Loading…
Reference in New Issue
Block a user