summaryrefslogtreecommitdiffstats
path: root/stream.c
diff options
context:
space:
mode:
authorKaz Kylheku <kaz@kylheku.com>2016-03-19 06:43:59 -0700
committerKaz Kylheku <kaz@kylheku.com>2016-03-19 06:43:59 -0700
commitd23a45711a9c6864f7c4adf962499fe3d1a4adb5 (patch)
tree5a8e0fe23cf069149181b52a772038333bf280ec /stream.c
parent54ab368d20df1796106e2dae222a3b926c331dd9 (diff)
downloadtxr-d23a45711a9c6864f7c4adf962499fe3d1a4adb5.tar.gz
txr-d23a45711a9c6864f7c4adf962499fe3d1a4adb5.tar.bz2
txr-d23a45711a9c6864f7c4adf962499fe3d1a4adb5.zip
Buffer size digit in file open mode string.
* streamn.c (struct stdio_handle): New member, buf. (stdio_stream_destroy): Free the stdio_handle's buf. (parse_mode): Handle digit character, converting it to integer value stored in m.buforder. (set_mode_props): Allocate a buffer and install into FILE * stream if the mode specifies a buforder. (make_stdio_stream_common): Initialize buffer to null. * stream.h (struct stdio_mode): New signed bitfield member, buforder. (stdio_mode_init_trivial): Initialize buforder member to -1. * txr.1: Documented size order digit.
Diffstat (limited to 'stream.c')
-rw-r--r--stream.c27
1 files changed, 25 insertions, 2 deletions
diff --git a/stream.c b/stream.c
index 5798efcb..ad83fc99 100644
--- a/stream.c
+++ b/stream.c
@@ -355,6 +355,7 @@ struct stdio_handle {
val unget_c;
utf8_decoder_t ud;
val err;
+ char *buf;
#if HAVE_FORK_STUFF
pid_t pid;
#else
@@ -391,6 +392,7 @@ static void stdio_stream_destroy(val stream)
struct stdio_handle *h = coerce(struct stdio_handle *, stream->co.handle);
close_stream(stream, nil);
strm_base_cleanup(&h->a);
+ free(h->buf);
free(h);
}
@@ -1200,6 +1202,14 @@ static struct stdio_mode parse_mode(val mode_str)
}
m.unbuf = 1;
break;
+ case '0': case '1': case '2': case '3': case '4':
+ case '5': case '6': case '7': case '8': case '9':
+ if (m.unbuf) {
+ m.malformed = 1;
+ return m;
+ }
+ m.buforder = *ms - '0';
+ break;
default:
m.malformed = 1;
return m;
@@ -1259,14 +1269,26 @@ val normalize_mode(struct stdio_mode *m, val mode_str)
val set_mode_props(const struct stdio_mode m, val stream)
{
- if (m.interactive || m.linebuf || m.unbuf) {
+ if (m.interactive || m.linebuf || m.unbuf || m.buforder != -1) {
struct stdio_handle *h = coerce(struct stdio_handle *,
cobj_handle(stream, stdio_stream_s));
if (h->f) {
+ int size = m.buforder == -1 ? 0 : 1024 << m.buforder;
+
+ if (h->buf) {
+ free(h->buf);
+ h->buf = 0;
+ }
+
+ if (size)
+ h->buf = coerce(char *, chk_malloc(size));
+
if (m.write && (m.linebuf || (m.interactive && !m.unbuf)))
- setvbuf(h->f, 0, _IOLBF, 0);
+ setvbuf(h->f, h->buf, _IOLBF, size);
else if (m.unbuf)
setbuf(h->f, 0);
+ else if (size)
+ setvbuf(h->f, h->buf, _IOFBF, size);
}
if (m.interactive)
@@ -1285,6 +1307,7 @@ static val make_stdio_stream_common(FILE *f, val descr, struct cobj_ops *ops)
h->unget_c = nil;
utf8_decoder_init(&h->ud);
h->err = nil;
+ h->buf = 0;
h->pid = 0;
h->mode = nil;
h->is_rotated = 0;