summaryrefslogtreecommitdiffstats
path: root/stream.c
diff options
context:
space:
mode:
authorKaz Kylheku <kaz@kylheku.com>2016-05-28 13:14:28 -0700
committerKaz Kylheku <kaz@kylheku.com>2016-05-28 13:14:28 -0700
commit42367d8856e5476245b218e1b2bbf416a0341327 (patch)
treef1da85ccabd1c670c44c916c744b63c4064dbef6 /stream.c
parent0c2ccd85a2a237c442f0706e420797af64a4398f (diff)
downloadtxr-42367d8856e5476245b218e1b2bbf416a0341327.tar.gz
txr-42367d8856e5476245b218e1b2bbf416a0341327.tar.bz2
txr-42367d8856e5476245b218e1b2bbf416a0341327.zip
Add a byte mode to stdio streams.
* stream.c (byte_oriented_k): New keyword symbol variable. (struct stdio_handle): New member, is_byte_oriented. (stdio_get_prop): Retrieve the value of is_byte_oriented when the :byte-oriented property is inquired. (stdio_set_prop): Map :byte-oriented to the is_byte_oriented flag. (stdio_get_char): Do not decode UTF-8 if is_byte_oriented is set; just read one character. (make_stdio_stream_common): Initialize is_byte_oriented to 0. (stream_init): Initialize byte_oriented_k. * txr.1: Document :byte-oriented property, along with some clean-up and clarification in the description of properties.
Diffstat (limited to 'stream.c')
-rw-r--r--stream.c26
1 files changed, 22 insertions, 4 deletions
diff --git a/stream.c b/stream.c
index 469416be..d8babd36 100644
--- a/stream.c
+++ b/stream.c
@@ -71,7 +71,7 @@ val print_flo_precision_s, print_flo_digits_s, print_flo_format_s;
val print_base_s;
val from_start_k, from_current_k, from_end_k;
-val real_time_k, name_k, addr_k, fd_k;
+val real_time_k, name_k, addr_k, fd_k, byte_oriented_k;
val format_s;
val stdio_stream_s;
@@ -364,6 +364,7 @@ struct stdio_handle {
val mode; /* used by tail */
unsigned is_rotated; /* used by tail */
unsigned is_real_time;
+ unsigned is_byte_oriented;
#if HAVE_SOCKETS
val family;
val type;
@@ -579,16 +580,22 @@ static val stdio_get_prop(val stream, val ind)
return h->descr;
} else if (ind == fd_k) {
return h->f ? num(fileno(h->f)) : nil;
+ } else if (ind == byte_oriented_k) {
+ return h->is_byte_oriented ? t : nil;
}
return nil;
}
static val stdio_set_prop(val stream, val ind, val prop)
{
+ struct stdio_handle *h = coerce(struct stdio_handle *, stream->co.handle);
+
if (ind == real_time_k) {
- struct stdio_handle *h = coerce(struct stdio_handle *, stream->co.handle);
h->is_real_time = prop ? 1 : 0;
return t;
+ } else if (ind == byte_oriented_k) {
+ h->is_byte_oriented = prop ? 1 : 0;
+ return t;
}
return nil;
}
@@ -688,8 +695,17 @@ static val stdio_get_char(val stream)
return rcyc_pop(&h->unget_c);
if (h->f) {
- wint_t ch = utf8_decode(&h->ud, stdio_get_char_callback,
- coerce(mem_t *, h->f));
+ wint_t ch;
+
+ if (h->is_byte_oriented) {
+ ch = se_getc(h->f);
+ if (ch == 0)
+ ch = 0xDC00;
+ } else {
+ ch = utf8_decode(&h->ud, stdio_get_char_callback,
+ coerce(mem_t *, h->f));
+ }
+
return (ch != WEOF) ? chr(ch) : stdio_maybe_read_error(stream);
}
return stdio_maybe_read_error(stream);
@@ -1307,6 +1323,7 @@ static val make_stdio_stream_common(FILE *f, val descr, struct cobj_ops *ops)
#else
h->is_real_time = 0;
#endif
+ h->is_byte_oriented = 0;
#if HAVE_SOCKETS
h->family = nil;
h->type = nil;
@@ -3898,6 +3915,7 @@ void stream_init(void)
name_k = intern(lit("name"), keyword_package);
addr_k = intern(lit("addr"), keyword_package);
fd_k = intern(lit("fd"), keyword_package);
+ byte_oriented_k = intern(lit("byte-oriented"), keyword_package);
format_s = intern(lit("format"), user_package);
stdio_stream_s = intern(lit("stdio-stream"), user_package);
#if HAVE_SOCKETS