summaryrefslogtreecommitdiffstats
path: root/stream.c
diff options
context:
space:
mode:
authorKaz Kylheku <kaz@kylheku.com>2017-06-05 22:39:10 -0700
committerKaz Kylheku <kaz@kylheku.com>2017-06-05 22:39:10 -0700
commit9bc7cb70ae8d216f90f11ba66e202e941c469a91 (patch)
tree8f64edba42b9d2074604a4970b134fd90ac9e8df /stream.c
parentfa2318f5fb705f106bf251e2dd52f22c5c6bc5f4 (diff)
downloadtxr-9bc7cb70ae8d216f90f11ba66e202e941c469a91.tar.gz
txr-9bc7cb70ae8d216f90f11ba66e202e941c469a91.tar.bz2
txr-9bc7cb70ae8d216f90f11ba66e202e941c469a91.zip
buffers: improve put-buf and fill-buf.
* stream.h (struct strm_ops): put_buf and fill_buf function pointers get third argument to indicate starting position of read and write. (put_buf, fill_buf): Declarations updated. * stream.c (unimpl_put_buf, unimpl_fill_buf): Third argument added. (generic_put_buf, generic_fill_buf, stdio_put_buf, stdio_fill_buf): Implement position argument. (delegate_put_buf, delegate_fill_buf): Take third argument, pass it down. (put_buf, fill_buf): New position argument in second position. Defaulted to zero. Passed down. (stream_init): Updated registration of put-buf and fill-buf. * txr.1: Updated documentation.
Diffstat (limited to 'stream.c')
-rw-r--r--stream.c75
1 files changed, 48 insertions, 27 deletions
diff --git a/stream.c b/stream.c
index 0b2e607a..a4560f05 100644
--- a/stream.c
+++ b/stream.c
@@ -179,12 +179,12 @@ static noreturn val unimpl_unget_byte(val stream, int byte)
unimpl(stream, lit("unget-byte"));
}
-static noreturn val unimpl_put_buf(val stream, val buf)
+static noreturn val unimpl_put_buf(val stream, val buf, cnum pos)
{
unimpl(stream, lit("put-buf"));
}
-static noreturn val unimpl_fill_buf(val stream, val buf)
+static noreturn val unimpl_fill_buf(val stream, val buf, cnum pos)
{
unimpl(stream, lit("fill-buf"));
}
@@ -304,33 +304,42 @@ static val null_get_fd(val stream)
return nil;
}
-static val generic_put_buf(val stream, val buf)
+static val generic_put_buf(val stream, val buf, cnum pos)
{
val self = lit("put-buf");
struct strm_ops *ops = coerce(struct strm_ops *, stream->co.ops);
cnum len = c_num(length_buf(buf)), i;
mem_t *ptr = buf_get(buf, self);
- for (i = 0; i < len; i++)
+ if (pos >= len)
+ return num(len);
+
+ for (i = pos; i < len; i++)
ops->put_byte(stream, *ptr++);
- return t;
+ if (i > len)
+ i = len;
+
+ return num(i);
}
-static val generic_fill_buf(val stream, val buf)
+static val generic_fill_buf(val stream, val buf, cnum pos)
{
val self = lit("fill-buf");
struct strm_ops *ops = coerce(struct strm_ops *, stream->co.ops);
cnum len = c_num(length_buf(buf)), i;
mem_t *ptr = buf_get(buf, self);
- for (i = 0; i < len; i++) {
+ for (i = pos; i < len; i++) {
val byte = ops->get_byte(stream);
if (!byte)
break;
*ptr++ = c_num(byte);
}
+ if (i > len)
+ i = len;
+
return num(i);
}
@@ -845,7 +854,7 @@ static val stdio_unget_byte(val stream, int byte)
: stdio_maybe_error(stream, lit("writing"));
}
-static val stdio_put_buf(val stream, val buf)
+static val stdio_put_buf(val stream, val buf, cnum pos)
{
val self = lit("put-buf");
ucnum len = c_unum(length_buf(buf));
@@ -853,26 +862,36 @@ static val stdio_put_buf(val stream, val buf)
struct stdio_handle *h = coerce(struct stdio_handle *, stream->co.handle);
if ((size_t) len != len)
uw_throwf(error_s, lit("~a: buffer too large"), self, nao);
+ if (pos >= len)
+ return num(len);
errno = 0;
- return h->f != 0 && fwrite(ptr, 1, len, h->f) == len
- ? t : stdio_maybe_error(stream, lit("writing"));
+ if (h->f != 0) {
+ cnum nwrit = fwrite(ptr + pos, 1, len - pos, h->f);
+ if (nwrit > 0)
+ return num(pos + nwrit);
+ }
+ stdio_maybe_error(stream, lit("writing"));
+ return zero;
}
-static val stdio_fill_buf(val stream, val buf)
+static val stdio_fill_buf(val stream, val buf, cnum pos)
{
val self = lit("fill-buf");
ucnum len = c_unum(length_buf(buf));
mem_t *ptr = buf_get(buf, self);
struct stdio_handle *h = coerce(struct stdio_handle *, stream->co.handle);
- size_t nread = 0;
if ((size_t) len != len)
uw_throwf(error_s, lit("~a: buffer too large"), self, nao);
+ if (pos >= len)
+ return num(len);
errno = 0;
- if (h->f != 0)
- nread = fread(ptr, 1, len, h->f);
- if (nread == len || nread < len || feof(h->f))
- return unum(len);
- return stdio_maybe_read_error(stream);
+ if (h->f != 0) {
+ cnum nread = fread(ptr + pos, 1, len - pos, h->f);
+ if (nread > 0)
+ return unum(pos + nread);
+ }
+ stdio_maybe_read_error(stream);
+ return zero;
}
static val stdio_close(val stream, val throw_on_error)
@@ -2518,16 +2537,16 @@ static val delegate_unget_byte(val stream, int byte)
return s->target_ops->unget_byte(s->target_stream, byte);
}
-static val delegate_put_buf(val stream, val buf)
+static val delegate_put_buf(val stream, val buf, cnum pos)
{
struct delegate_base *s = coerce(struct delegate_base *, stream->co.handle);
- return s->target_ops->put_buf(s->target_stream, buf);
+ return s->target_ops->put_buf(s->target_stream, buf, pos);
}
-static val delegate_fill_buf(val stream, val buf)
+static val delegate_fill_buf(val stream, val buf, cnum pos)
{
struct delegate_base *s = coerce(struct delegate_base *, stream->co.handle);
- return s->target_ops->fill_buf(s->target_stream, buf);
+ return s->target_ops->fill_buf(s->target_stream, buf, pos);
}
static val delegate_close(val stream, val throw_on_error)
@@ -2787,18 +2806,20 @@ val unget_byte(val byte, val stream_in)
return ops->unget_byte(stream, b);
}
-val put_buf(val buf, val stream_in)
+val put_buf(val buf, val pos_in, val stream_in)
{
val stream = default_arg(stream_in, std_output);
+ cnum pos = c_num(default_arg(pos_in, zero));
struct strm_ops *ops = coerce(struct strm_ops *, cobj_ops(stream, stream_s));
- return ops->put_buf(stream, buf);
+ return ops->put_buf(stream, buf, pos);
}
-val fill_buf(val buf, val stream_in)
+val fill_buf(val buf, val pos_in, val stream_in)
{
val stream = default_arg(stream_in, std_input);
+ cnum pos = c_num(default_arg(pos_in, zero));
struct strm_ops *ops = coerce(struct strm_ops *, cobj_ops(stream, stream_s));
- return ops->fill_buf(stream, buf);
+ return ops->fill_buf(stream, buf, pos);
}
struct fmt {
@@ -4426,8 +4447,8 @@ void stream_init(void)
reg_fun(intern(lit("put-strings"), user_package), func_n2o(put_strings, 1));
reg_fun(intern(lit("unget-char"), user_package), func_n2o(unget_char, 1));
reg_fun(intern(lit("unget-byte"), user_package), func_n2o(unget_byte, 1));
- reg_fun(intern(lit("put-buf"), user_package), func_n2o(put_buf, 1));
- reg_fun(intern(lit("fill-buf"), user_package), func_n2o(fill_buf, 1));
+ reg_fun(intern(lit("put-buf"), user_package), func_n3o(put_buf, 1));
+ reg_fun(intern(lit("fill-buf"), user_package), func_n3o(fill_buf, 1));
reg_fun(intern(lit("flush-stream"), user_package), func_n1o(flush_stream, 0));
reg_fun(intern(lit("seek-stream"), user_package), func_n3(seek_stream));
reg_fun(intern(lit("truncate-stream"), user_package), func_n2(truncate_stream));