summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
-rw-r--r--include/stdio.h4
-rw-r--r--src/internal/stdio_impl.h2
-rw-r--r--src/stdio/fclose.c1
-rw-r--r--src/stdio/fgetln.c19
4 files changed, 25 insertions, 1 deletions
diff --git a/include/stdio.h b/include/stdio.h
index 3d22220f..19ab8bcd 100644
--- a/include/stdio.h
+++ b/include/stdio.h
@@ -172,6 +172,10 @@ int getw(FILE *);
int putw(int, FILE *);
#endif
+#ifdef _BSD_SOURCE
+char *fgetln(FILE *, size_t *);
+#endif
+
#ifdef _GNU_SOURCE
int asprintf(char **, const char *, ...);
int vasprintf(char **, const char *, va_list);
diff --git a/src/internal/stdio_impl.h b/src/internal/stdio_impl.h
index d54c918b..65dcfbda 100644
--- a/src/internal/stdio_impl.h
+++ b/src/internal/stdio_impl.h
@@ -57,7 +57,7 @@ struct __FILE_s {
int waiters;
void *cookie;
off_t off;
- void *dummy4;
+ char *getln_buf;
void *mustbezero_2;
unsigned char *shend;
off_t shlim, shcnt;
diff --git a/src/stdio/fclose.c b/src/stdio/fclose.c
index 373a2c76..8fdc3f7d 100644
--- a/src/stdio/fclose.c
+++ b/src/stdio/fclose.c
@@ -16,6 +16,7 @@ int fclose(FILE *f)
r = fflush(f);
r |= f->close(f);
+ if (f->getln_buf) free(f->getln_buf);
if (!perm) free(f);
return r;
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;
+}