summaryrefslogtreecommitdiffhomepage
path: root/common-channel.c
diff options
context:
space:
mode:
Diffstat (limited to 'common-channel.c')
-rw-r--r--common-channel.c68
1 files changed, 44 insertions, 24 deletions
diff --git a/common-channel.c b/common-channel.c
index 6dce497..5f4051e 100644
--- a/common-channel.c
+++ b/common-channel.c
@@ -434,10 +434,36 @@ static void send_msg_channel_eof(struct Channel *channel) {
TRACE(("leave send_msg_channel_eof"))
}
-/* Called to write data out to the local side of the channel.
- Writes the circular buffer contents and also the "moredata" buffer
- if not null. Will ignore EAGAIN */
-static void writechannel(struct Channel* channel, int fd, circbuffer *cbuf,
+#ifndef HAVE_WRITEV
+static void writechannel_fallback(struct Channel* channel, int fd, circbuffer *cbuf,
+ const unsigned char *UNUSED(moredata), unsigned int *morelen) {
+
+ unsigned char *circ_p1, *circ_p2;
+ unsigned int circ_len1, circ_len2;
+ ssize_t written;
+
+ if (morelen) {
+ /* fallback doesn't consume moredata */
+ *morelen = 0;
+ }
+
+ /* Write the first portion of the circular buffer */
+ cbuf_readptrs(cbuf, &circ_p1, &circ_len1, &circ_p2, &circ_len2);
+ written = write(fd, circ_p1, circ_len1);
+ if (written < 0) {
+ if (errno != EINTR && errno != EAGAIN) {
+ TRACE(("channel IO write error fd %d %s", fd, strerror(errno)))
+ close_chan_fd(channel, fd, SHUT_WR);
+ }
+ } else {
+ cbuf_incrread(cbuf, written);
+ channel->recvdonelen += written;
+ }
+}
+#endif /* !HAVE_WRITEV */
+
+#ifdef HAVE_WRITEV
+static void writechannel_writev(struct Channel* channel, int fd, circbuffer *cbuf,
const unsigned char *moredata, unsigned int *morelen) {
struct iovec iov[3];
@@ -445,9 +471,7 @@ static void writechannel(struct Channel* channel, int fd, circbuffer *cbuf,
unsigned int circ_len1, circ_len2;
int io_count = 0;
- int written;
-
- TRACE(("enter writechannel fd %d", fd))
+ ssize_t written;
cbuf_readptrs(cbuf, &circ_p1, &circ_len1, &circ_p2, &circ_len2);
@@ -502,24 +526,19 @@ static void writechannel(struct Channel* channel, int fd, circbuffer *cbuf,
channel->recvdonelen += written;
}
-#if 0
-
- maxlen = cbuf_readlen(cbuf);
-
- /* Write the data out */
- len = write(fd, cbuf_readptr(cbuf, maxlen), maxlen);
- if (len <= 0) {
- TRACE(("errno %d len %d", errno, len))
- if (len < 0 && errno != EINTR) {
- close_chan_fd(channel, fd, SHUT_WR);
- }
- TRACE(("leave writechannel: len <= 0"))
- return;
- }
- TRACE(("writechannel wrote %d", len))
+}
+#endif /* HAVE_WRITEV */
- cbuf_incrread(cbuf, len);
- channel->recvdonelen += len;
+/* Called to write data out to the local side of the channel.
+ Writes the circular buffer contents and also the "moredata" buffer
+ if not null. Will ignore EAGAIN */
+static void writechannel(struct Channel* channel, int fd, circbuffer *cbuf,
+ const unsigned char *moredata, unsigned int *morelen) {
+ TRACE(("enter writechannel fd %d", fd))
+#ifdef HAVE_WRITEV
+ writechannel_writev(channel, fd, cbuf, moredata, morelen);
+#else
+ writechannel_fallback(channel, fd, cbuf, moredata, morelen);
#endif
/* Window adjust handling */
@@ -537,6 +556,7 @@ static void writechannel(struct Channel* channel, int fd, circbuffer *cbuf,
TRACE(("leave writechannel"))
}
+
/* Set the file descriptors for the main select in session.c
* This avoid channels which don't have any window available, are closed, etc*/
void setchannelfds(fd_set *readfds, fd_set *writefds, int allow_reads) {