diff options
author | Rich Felker <dalias@aerifal.cx> | 2012-08-11 18:10:38 -0400 |
---|---|---|
committer | Rich Felker <dalias@aerifal.cx> | 2012-08-11 18:10:38 -0400 |
commit | 617182734ca0beffa347747019d78b972e2038f9 (patch) | |
tree | ef09b631f0057e571e7ea707c281b68bdc60503c /src/stdio/fgetln.c | |
parent | 2b964b010e5d37cb2ff712d57a14095188d689e3 (diff) | |
download | musl-617182734ca0beffa347747019d78b972e2038f9.tar.gz musl-617182734ca0beffa347747019d78b972e2038f9.tar.bz2 musl-617182734ca0beffa347747019d78b972e2038f9.tar.xz musl-617182734ca0beffa347747019d78b972e2038f9.zip |
add bsd fgetln function
optimized to avoid allocation and return lines directly out of the
stream buffer whenever possible.
Diffstat (limited to 'src/stdio/fgetln.c')
-rw-r--r-- | src/stdio/fgetln.c | 19 |
1 files changed, 19 insertions, 0 deletions
diff --git a/src/stdio/fgetln.c b/src/stdio/fgetln.c new file mode 100644 index 00000000..06b88837 --- /dev/null +++ b/src/stdio/fgetln.c @@ -0,0 +1,19 @@ +#include "stdio_impl.h" + +char *fgetln(FILE *f, size_t *plen) +{ + char *ret = 0, *z; + ssize_t l; + FLOCK(f); + ungetc(getc_unlocked(f), f); + if ((z=memchr(f->rpos, '\n', f->rend - f->rpos))) { + ret = (char *)f->rpos; + *plen = ++z - ret; + f->rpos = (void *)z; + } else if ((l = getline(&f->getln_buf, (size_t[]){0}, f)) > 0) { + *plen = l; + ret = f->getln_buf; + } + FUNLOCK(f); + return ret; +} |