summaryrefslogtreecommitdiffstats
path: root/stream.c
diff options
context:
space:
mode:
authorKaz Kylheku <kaz@kylheku.com>2016-03-19 09:39:14 -0700
committerKaz Kylheku <kaz@kylheku.com>2016-03-19 09:39:14 -0700
commit8d338fb27ee545c4f2fc90963f8d0c70af478b66 (patch)
tree3f01780645e6407e904e46be74565b7d77832073 /stream.c
parentea9ba58da58fc03e4b1ae1d6e277dad8189fa920 (diff)
downloadtxr-8d338fb27ee545c4f2fc90963f8d0c70af478b66.tar.gz
txr-8d338fb27ee545c4f2fc90963f8d0c70af478b66.tar.bz2
txr-8d338fb27ee545c4f2fc90963f8d0c70af478b66.zip
Permissive stream open mode strings.
There is more to this patch than just more permissive mode strings. Now if a socket can be opened with mode "l2" for instance, and these options are effectively applied to the socket-specific "r+b" default, not to "r". * stream.c (parse_mode): New argument specifying a default mode. The syntax is relaxed, allowing previously required elements to be omitted. (normalize_mode): New argument specifying a default mode. Format mode is always called now, because an input string is no longer necessarily a valid fopen string even in cases when it doesn't specify any extensions. (open_file, open_fileno, open_tail, open_command, open_process): Use new normalize_mode argument for defaulting; normalize_mode no longer defaults to "r". * stream.h (stdio_mode_init_trivial): Macro removed. (stdio_mode_init_blank, stdio_mode_init_r, stdio_mode_init_rpb): New initializer macros. (parse_mode, normalize_mode): Declarations updated. * socket.c (sock_accept): In datagram socket case, use new parse_mode argument for defaulting using stdio_mode_init_rpb, rather than overriding a missing string with "r+b". (open_sockfd): Likewise, and use new normalize_mode argument similarly for defaulting the mode on a stream socket. * txr.1: Documented mode string permissiveness.
Diffstat (limited to 'stream.c')
-rw-r--r--stream.c50
1 files changed, 22 insertions, 28 deletions
diff --git a/stream.c b/stream.c
index 9029ddc1..713671cb 100644
--- a/stream.c
+++ b/stream.c
@@ -1148,9 +1148,9 @@ static struct strm_ops pipe_ops =
stdio_clear_error,
stdio_get_fd);
-struct stdio_mode parse_mode(val mode_str)
+struct stdio_mode parse_mode(val mode_str, struct stdio_mode m_dfl)
{
- struct stdio_mode m = stdio_mode_init_trivial(0);
+ struct stdio_mode m = stdio_mode_init_blank;
const wchar_t *ms = c_str(mode_str);
switch (*ms) {
@@ -1169,8 +1169,7 @@ struct stdio_mode parse_mode(val mode_str)
m.append = 1;
break;
default:
- m.malformed = 1;
- return m;
+ break;
}
if (*ms == '+') {
@@ -1180,6 +1179,9 @@ struct stdio_mode parse_mode(val mode_str)
m.read = 1;
}
+ if (!m.read && !m.write)
+ m = m_dfl;
+
for (; *ms; ms++) {
switch (*ms) {
case 'b':
@@ -1247,24 +1249,16 @@ static val format_mode(const struct stdio_mode m)
return string(buf);
}
-val normalize_mode(struct stdio_mode *m, val mode_str)
+val normalize_mode(struct stdio_mode *m, val mode_str_in, struct stdio_mode m_dfl)
{
- struct stdio_mode blank = stdio_mode_init_trivial(1);
+ val mode_str = default_arg(mode_str_in, lit(""));
- if (null_or_missing_p(mode_str)) {
- *m = blank;
- return lit("r");
- } else {
- *m = parse_mode(mode_str);
+ *m = parse_mode(mode_str, m_dfl);
- if (m->malformed)
- uw_throwf(file_error_s, lit("invalid file open mode ~a"), mode_str, nao);
+ if (m->malformed)
+ uw_throwf(file_error_s, lit("invalid file open mode ~a"), mode_str, nao);
- if (!m->interactive)
- return mode_str;
-
- return format_mode(*m);
- }
+ return format_mode(*m);
}
val set_mode_props(const struct stdio_mode m, val stream)
@@ -3358,8 +3352,8 @@ val open_directory(val path)
val open_file(val path, val mode_str)
{
- struct stdio_mode m;
- FILE *f = w_fopen(c_str(path), c_str(normalize_mode(&m, mode_str)));
+ struct stdio_mode m, m_r = stdio_mode_init_r;
+ FILE *f = w_fopen(c_str(path), c_str(normalize_mode(&m, mode_str, m_r)));
if (!f)
uw_throwf(file_error_s, lit("error opening ~a: ~d/~s"),
@@ -3370,8 +3364,8 @@ val open_file(val path, val mode_str)
val open_fileno(val fd, val mode_str)
{
- struct stdio_mode m;
- FILE *f = (errno = 0, w_fdopen(c_num(fd), c_str(normalize_mode(&m, mode_str))));
+ struct stdio_mode m, m_r = stdio_mode_init_r;
+ FILE *f = (errno = 0, w_fdopen(c_num(fd), c_str(normalize_mode(&m, mode_str, m_r))));
if (!f) {
close(c_num(fd));
@@ -3386,8 +3380,8 @@ val open_fileno(val fd, val mode_str)
val open_tail(val path, val mode_str, val seek_end_p)
{
- struct stdio_mode m;
- val mode = normalize_mode(&m, mode_str);
+ struct stdio_mode m, m_r = stdio_mode_init_r;
+ val mode = normalize_mode(&m, mode_str, m_r);
FILE *f = w_fopen(c_str(path), c_str(mode));
struct stdio_handle *h;
val stream;
@@ -3408,8 +3402,8 @@ val open_tail(val path, val mode_str, val seek_end_p)
val open_command(val path, val mode_str)
{
- struct stdio_mode m;
- FILE *f = w_popen(c_str(path), c_str(normalize_mode(&m, mode_str)));
+ struct stdio_mode m, m_r = stdio_mode_init_r;
+ FILE *f = w_popen(c_str(path), c_str(normalize_mode(&m, mode_str, m_r)));
if (!f)
uw_throwf(file_error_s, lit("error opening pipe ~a: ~d/~s"),
@@ -3421,8 +3415,8 @@ val open_command(val path, val mode_str)
#if HAVE_FORK_STUFF
val open_process(val name, val mode_str, val args)
{
- struct stdio_mode m;
- val mode = normalize_mode(&m, mode_str);
+ struct stdio_mode m, m_r = stdio_mode_init_r;
+ val mode = normalize_mode(&m, mode_str, m_r);
int input = m.read != 0;
int fd[2];
pid_t pid;