diff options
author | Kaz Kylheku <kaz@kylheku.com> | 2016-03-19 06:54:14 -0700 |
---|---|---|
committer | Kaz Kylheku <kaz@kylheku.com> | 2016-03-19 06:54:14 -0700 |
commit | ea9ba58da58fc03e4b1ae1d6e277dad8189fa920 (patch) | |
tree | dca22ce43dcfaf3489ddf33513b94ab41b786655 | |
parent | 5e3c8d5663f7c7f001669bb9d91485244f8ce799 (diff) | |
download | txr-ea9ba58da58fc03e4b1ae1d6e277dad8189fa920.tar.gz txr-ea9ba58da58fc03e4b1ae1d6e277dad8189fa920.tar.bz2 txr-ea9ba58da58fc03e4b1ae1d6e277dad8189fa920.zip |
Size mode meaningful in datagram sockets.
* socket.c (struct dgram_stream): new rx_max member.
(make_dgram_sock_stream): New arguments: a struct stdio_mode,
and pointer to prototype dgram socket. If a size is specified
in the mode, then use that as rx_max. Otherwise if a prototype
socket is specified, use its rx_max as the new socket's rx_max.
Otherwise default on 65536.
(dgram_get_byte_callback): Use d->rx_max as the capture size,
rather than a hard-coded 65536.
(sock_accept): Use d->rx_max as capture size for datagram.
Parse the mode. Pass the parsed mode to make_dgram_sock_stream,
as well as the accepting socket, so it can set up the rx_max
for the new socket.
(open_sockfd): Parse the mode and pass to make_dgram_sock_stream.
* stream.c (parse_mode): Static function becomes extern.
* stream.h (parse_mode): Declared.
* txr.1: Documented.
-rw-r--r-- | socket.c | 31 | ||||
-rw-r--r-- | stream.c | 2 | ||||
-rw-r--r-- | stream.h | 1 | ||||
-rw-r--r-- | txr.1 | 26 |
4 files changed, 48 insertions, 12 deletions
@@ -65,7 +65,7 @@ struct dgram_stream { int err; mem_t *rx_buf; mem_t *tx_buf; - int rx_size, rx_pos; + int rx_max, rx_size, rx_pos; int tx_pos; unsigned sock_connected : 1; }; @@ -261,7 +261,8 @@ static_forward(struct strm_ops dgram_strm_ops); static val make_dgram_sock_stream(int fd, val family, val peer, mem_t *dgram, int dgram_size, - struct sockaddr *peer_addr, socklen_t pa_len) + struct sockaddr *peer_addr, socklen_t pa_len, + struct stdio_mode m, struct dgram_stream *d_proto) { struct dgram_stream *d = coerce(struct dgram_stream *, chk_malloc(sizeof *d)); @@ -273,6 +274,9 @@ static val make_dgram_sock_stream(int fd, val family, val peer, d->family = d->peer = d->addr = d->unget_c = nil; d->err = 0; d->rx_buf = dgram; + d->rx_max = (m.buforder == -1) + ? d_proto ? d_proto->rx_max : 65536 + : 1024 << m.buforder; d->rx_size = dgram_size; d->rx_pos = 0; d->tx_buf = 0; @@ -373,7 +377,7 @@ static int dgram_get_byte_callback(mem_t *ctx) if (d->rx_buf) { return (d->rx_pos < d->rx_size) ? d->rx_buf[d->rx_pos++] : EOF; } else { - const int dgram_size = 65536; + const int dgram_size = d->rx_max; mem_t *dgram = chk_malloc(dgram_size); ssize_t nbytes = -1; @@ -762,7 +766,7 @@ failed: num(errno), string_utf8(strerror(errno)), nao); } -static val sock_accept(val sock, val mode_str, val timeout_in) +static val sock_accept(val sock, val mode_str_in, val timeout_in) { val sfd = stream_fd(sock); int fd = sfd ? c_num(sfd) : -1; @@ -804,8 +808,9 @@ static val sock_accept(val sock, val mode_str, val timeout_in) } if (type == num_fast(SOCK_DGRAM)) { + struct dgram_stream *d = coerce(struct dgram_stream *, sock->co.handle); ssize_t nbytes = -1; - const int dgram_size = 65536; + const int dgram_size = d->rx_max; mem_t *dgram = chk_malloc(dgram_size); if (sock_peer(sock)) { @@ -845,7 +850,9 @@ static val sock_accept(val sock, val mode_str, val timeout_in) { int afd = dup(fd); + val mode_str = default_arg(mode_str_in, lit("r+b")); mem_t *shrink = chk_realloc(dgram, nbytes); + if (shrink) dgram = shrink; @@ -855,9 +862,9 @@ static val sock_accept(val sock, val mode_str, val timeout_in) uw_throwf(socket_error_s, lit("sock-accept: unable to "), family, nao); } - return make_dgram_sock_stream(afd, family, peer, dgram, nbytes, - coerce(struct sockaddr *, &sa), salen); + coerce(struct sockaddr *, &sa), salen, + parse_mode(mode_str), d); } } else { int afd = -1; @@ -881,7 +888,7 @@ static val sock_accept(val sock, val mode_str, val timeout_in) family, nao); { - val stream = open_sockfd(num(afd), family, num_fast(SOCK_STREAM), mode_str); + val stream = open_sockfd(num(afd), family, num_fast(SOCK_STREAM), mode_str_in); sock_set_peer(stream, peer); return stream; } @@ -939,11 +946,13 @@ static val sock_recv_timeout(val sock, val usec) val open_sockfd(val fd, val family, val type, val mode_str_in) { + struct stdio_mode m; + val mode_str = default_arg(mode_str_in, lit("r+b")); + if (type == num_fast(SOCK_DGRAM)) { - return make_dgram_sock_stream(c_num(fd), family, nil, 0, 0, 0, 0); + return make_dgram_sock_stream(c_num(fd), family, nil, 0, 0, 0, 0, + parse_mode(mode_str), 0); } else { - struct stdio_mode m; - val mode_str = default_arg(mode_str_in, lit("r+b")); FILE *f = (errno = 0, w_fdopen(c_num(fd), c_str(normalize_mode(&m, mode_str)))); if (!f) { @@ -1148,7 +1148,7 @@ static struct strm_ops pipe_ops = stdio_clear_error, stdio_get_fd); -static struct stdio_mode parse_mode(val mode_str) +struct stdio_mode parse_mode(val mode_str) { struct stdio_mode m = stdio_mode_init_trivial(0); const wchar_t *ms = c_str(mode_str); @@ -126,6 +126,7 @@ void fill_stream_ops(struct strm_ops *ops); void stream_print_op(val stream, val out, val pretty); void stream_mark_op(val stream); void stream_destroy_op(val stream); +struct stdio_mode parse_mode(val mode_str); val normalize_mode(struct stdio_mode *m, val mode_str); val set_mode_props(const struct stdio_mode m, val stream); val generic_get_line(val stream); @@ -37527,6 +37527,32 @@ Datagram servers should issue a new call should be issued for each client datagram, treating it as a new stream. +Datagram sockets ignore almost all aspects of the +.meta mode-string +passed in +.code open-socket +and +.codn sock-accept . +The only attribute not ignored is the buffer size specified +with a decimal digit character; however, it cannot be the +only item in the mode string. The string must be syntactically +valid, as described under the +.code open-file +function. The buffer size attribute controls the size used by +the datagram socket for receiving a datagram: the capture size. +A datagram socket has obtains a default capture size if one isn't +specified by the +.metn mode-string . +The default capture size is 65536 bytes for a datagram socket created by +.codn open-socket . +If a size is not passed to +.code sock-accept +via its +.meta mode-string +argument when it is invoked on a datagram socket, +that socket's size is used as the capture size of the +newly created datagram socket which is returned. + .coNP Structure @ sockaddr .synb .mets (defstruct sockaddr nil) |