|
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]> |
9 | 3 | *
|
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. |
21 | 7 | *
|
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. |
33 | 15 | */
|
34 | 16 |
|
35 |
| -#include <sys/types.h> |
| 17 | +/* |
| 18 | + * portable fgetln() version, NOT reentrant |
| 19 | + */ |
36 | 20 |
|
37 |
| -#include <errno.h> |
38 | 21 | #include <stdio.h>
|
39 | 22 | #include <stdlib.h>
|
40 |
| -#include <string.h> |
| 23 | +#include <errno.h> |
41 | 24 |
|
42 | 25 | #include "tmux.h"
|
43 | 26 |
|
44 | 27 | char *
|
45 | 28 | fgetln(FILE *fp, size_t *len)
|
46 | 29 | {
|
47 | 30 | 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; |
57 | 35 |
|
58 |
| - if (fgets(buf, bufsiz, fp) == NULL) |
| 36 | + if (!fp || !len) { |
| 37 | + errno = EINVAL; |
59 | 38 | 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))) |
71 | 42 | 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; |
80 | 44 | }
|
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; |
84 | 61 | }
|
85 |
| - |
0 commit comments