1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87
|
/*
Replace getline() with BSD specific fgetln().
Copyright (C) 2015 JianXiong Zhou <zhoujianxiong2@gmail.com>
Copyright (C) 2015 Jonathan Pelletier <funmungus(a)gmail.com>
This file is part of libscanmem.
This library is free software: you can redistribute it and/or modify
it under the terms of the GNU Lesser General Public License as published
by the Free Software Foundation; either version 3 of the License, or
(at your option) any later version.
This library is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU Lesser General Public License for more details.
You should have received a copy of the GNU Lesser General Public License
along with this library. If not, see <http://www.gnu.org/licenses/>.
*/
#include "getline.h"
#if !defined(HAVE_GETLINE) && defined(HAVE_FGETLN)
#include <stdlib.h>
#include <string.h>
#include <errno.h>
/*
* Avoid frequent malloc()/free() calls
* (determined by getline() test on Linux)
*/
#define BUF_MIN 120
ssize_t getline(char **lineptr, size_t *n, FILE *stream)
{
char *lptr;
size_t len = 0;
/* check for invalid arguments */
if (lineptr == NULL || n == NULL) {
errno = EINVAL;
return -1;
}
lptr = fgetln(stream, &len);
if (lptr == NULL) {
/* invalid stream */
errno = EINVAL;
return -1;
}
/*
* getline() returns a null byte ('\0') terminated C string,
* but fgetln() returns characters without '\0' termination
*/
if (*lineptr == NULL) {
*n = BUF_MIN;
goto alloc_buf;
}
/* realloc the original pointer */
if (*n < len + 1) {
free(*lineptr);
*n = len + 1;
alloc_buf:
*lineptr = malloc(*n);
if (*lineptr == NULL) {
*n = 0;
return -1;
}
}
/* copy over the string */
memcpy(*lineptr, lptr, len);
(*lineptr)[len] = '\0';
/*
* getline() and fgetln() both return len including the
* delimiter but without the null byte at the end
*/
return len;
}
#endif
|