Skip to content

Commit e9d369a

Browse files
committed
Fixed fgetln(3) implementation (from Joerg Jung) which does not depend on *BSD
fgets(3) semantics.
1 parent fc86452 commit e9d369a

File tree

1 file changed

+42
-66
lines changed

1 file changed

+42
-66
lines changed

compat/fgetln.c

+42-66
Original file line numberDiff line numberDiff line change
@@ -1,85 +1,61 @@
1-
/* $NetBSD: fgetln.c,v 1.3 2007/08/07 02:06:58 lukem Exp $ */
2-
3-
/*-
4-
* Copyright (c) 1998 The NetBSD Foundation, Inc.
5-
* All rights reserved.
6-
*
7-
* This code is derived from software contributed to The NetBSD Foundation
8-
* by Christos Zoulas.
1+
/*
2+
* Copyright (c) 2015 Joerg Jung <[email protected]>
93
*
10-
* Redistribution and use in source and binary forms, with or without
11-
* modification, are permitted provided that the following conditions
12-
* are met:
13-
* 1. Redistributions of source code must retain the above copyright
14-
* notice, this list of conditions and the following disclaimer.
15-
* 2. Redistributions in binary form must reproduce the above copyright
16-
* notice, this list of conditions and the following disclaimer in the
17-
* documentation and/or other materials provided with the distribution.
18-
* 3. Neither the name of The NetBSD Foundation nor the names of its
19-
* contributors may be used to endorse or promote products derived
20-
* from this software without specific prior written permission.
4+
* Permission to use, copy, modify, and distribute this software for any
5+
* purpose with or without fee is hereby granted, provided that the above
6+
* copyright notice and this permission notice appear in all copies.
217
*
22-
* THIS SOFTWARE IS PROVIDED BY THE NETBSD FOUNDATION, INC. AND CONTRIBUTORS
23-
* ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED
24-
* TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
25-
* PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE FOUNDATION OR CONTRIBUTORS
26-
* BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
27-
* CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
28-
* SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
29-
* INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
30-
* CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
31-
* ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
32-
* POSSIBILITY OF SUCH DAMAGE.
8+
* THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
9+
* WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
10+
* MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
11+
* ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
12+
* WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
13+
* ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
14+
* OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
3315
*/
3416

35-
#include <sys/types.h>
17+
/*
18+
* portable fgetln() version, NOT reentrant
19+
*/
3620

37-
#include <errno.h>
3821
#include <stdio.h>
3922
#include <stdlib.h>
40-
#include <string.h>
23+
#include <errno.h>
4124

4225
#include "tmux.h"
4326

4427
char *
4528
fgetln(FILE *fp, size_t *len)
4629
{
4730
static char *buf = NULL;
48-
static size_t bufsiz = 0;
49-
char *ptr;
50-
51-
52-
if (buf == NULL) {
53-
bufsiz = BUFSIZ;
54-
if ((buf = malloc(bufsiz)) == NULL)
55-
return NULL;
56-
}
31+
static size_t bufsz = 0;
32+
size_t r = 0;
33+
char *p;
34+
int c, e;
5735

58-
if (fgets(buf, bufsiz, fp) == NULL)
36+
if (!fp || !len) {
37+
errno = EINVAL;
5938
return NULL;
60-
61-
*len = 0;
62-
while ((ptr = strchr(&buf[*len], '\n')) == NULL) {
63-
size_t nbufsiz = bufsiz + BUFSIZ;
64-
char *nbuf = realloc(buf, nbufsiz);
65-
66-
if (nbuf == NULL) {
67-
int oerrno = errno;
68-
free(buf);
69-
errno = oerrno;
70-
buf = NULL;
39+
}
40+
if (!buf) {
41+
if (!(buf = calloc(1, BUFSIZ)))
7142
return NULL;
72-
} else
73-
buf = nbuf;
74-
75-
*len = bufsiz;
76-
if (fgets(&buf[bufsiz], BUFSIZ, fp) == NULL)
77-
return buf;
78-
79-
bufsiz = nbufsiz;
43+
bufsz = BUFSIZ;
8044
}
81-
82-
*len = (ptr - buf) + 1;
83-
return buf;
45+
while ((c = getc(fp)) != EOF) {
46+
buf[r++] = c;
47+
if (r == bufsz) {
48+
if (!(p = reallocarray(buf, 2, bufsz))) {
49+
e = errno;
50+
free(buf);
51+
errno = e;
52+
buf = NULL, bufsz = 0;
53+
return NULL;
54+
}
55+
buf = p, bufsz = 2 * bufsz;
56+
}
57+
if (c == '\n')
58+
break;
59+
}
60+
return (*len = r) ? buf : NULL;
8461
}
85-

0 commit comments

Comments
 (0)