Merged from silc_1_0_branch.
[silc.git] / lib / silcutil / unix / silcunixsockconn.c
index 330bab2dafa8631610875b698a913bc08dd36c5d..4d14ac5fc323626aa037aeff5eabedeca60441da 100644 (file)
@@ -33,6 +33,8 @@ int silc_socket_write(SilcSocketConnection sock)
   int fd = sock->sock;
   SilcBuffer src = sock->outbuf;
 
+  if (!src)
+    return -2;
   if (SILC_IS_DISABLED(sock))
     return -1;
 
@@ -41,14 +43,22 @@ int silc_socket_write(SilcSocketConnection sock)
   if (src->len > 0) {
     ret = write(fd, src->data, src->len);
     if (ret < 0) {
-      if (errno == EAGAIN) {
+      if (errno == EAGAIN || errno == EINTR) {
        SILC_LOG_DEBUG(("Could not write immediately, will do it later"));
        return -2;
       }
-      SILC_LOG_ERROR(("Cannot write to socket: %s", strerror(errno)));
+      SILC_LOG_DEBUG(("Cannot write to socket: %s", strerror(errno)));
+      sock->sock_error = errno;
       return -1;
     }
 
+    if (ret < src->len) {
+      SILC_LOG_DEBUG(("Wrote data %d of %d bytes, will write rest later",
+                     ret, src->len));
+      silc_buffer_pull(src, ret);
+      return -2;
+    }
+
     silc_buffer_pull(src, ret);
   }
 
@@ -57,6 +67,19 @@ int silc_socket_write(SilcSocketConnection sock)
   return ret;
 }
 
+/* QoS read handler, this will call the read and write events to indicate
+   that data is available again after a timeout. */
+
+SILC_TASK_CALLBACK(silc_socket_read_qos)
+{
+  SilcSocketConnection sock = context;
+  sock->qos->applied = TRUE;
+  silc_schedule_set_listen_fd(sock->qos->schedule, sock->sock,
+                             (SILC_TASK_READ | SILC_TASK_WRITE), TRUE);
+  sock->qos->applied = FALSE;
+  silc_socket_free(sock);
+}
+
 /* Reads data from the socket connection into the incoming data buffer.
    It reads as much as possible from the socket connection. This returns
    amount of bytes read or -1 on error or -2 on case where all of the
@@ -71,6 +94,43 @@ int silc_socket_read(SilcSocketConnection sock)
   if (SILC_IS_DISABLED(sock))
     return -1;
 
+  /* If QoS was applied to socket then return earlier read data but apply
+     QoS to it too, if necessary. */
+  if (sock->qos) {
+    if (sock->qos->applied) {
+      if (sock->qos->data_len) {
+       /* Pull hidden data since we have it from earlier QoS apply */
+       silc_buffer_pull_tail(sock->inbuf, sock->qos->data_len);
+       len = sock->qos->data_len;
+       sock->qos->data_len = 0;
+      }
+
+      if (sock->inbuf->len - len > sock->qos->read_limit_bytes) {
+       /* Seems we need to apply QoS for the remaining data as well */
+       silc_schedule_task_add(sock->qos->schedule, sock->sock,
+                              silc_socket_read_qos, silc_socket_dup(sock),
+                              sock->qos->limit_sec, sock->qos->limit_usec,
+                              SILC_TASK_TIMEOUT, SILC_TASK_PRI_LOW);
+       silc_schedule_unset_listen_fd(sock->qos->schedule, sock->sock);
+      
+       /* Hide the rest of the data from the buffer. */
+       sock->qos->data_len = (sock->inbuf->len - len - 
+                              sock->qos->read_limit_bytes);
+       silc_buffer_push_tail(sock->inbuf, sock->qos->data_len);
+      }
+
+      if (sock->inbuf->len)
+       return sock->inbuf->len;
+    }
+
+    /* If we were called and we have active QoS data pending, return
+       with no data */
+    if (sock->qos->data_len) {
+      silc_schedule_unset_listen_fd(sock->qos->schedule, sock->sock);
+      return -2;
+    }
+  }
+
   SILC_LOG_DEBUG(("Reading data from socket %d", fd));
 
   /* Read the data from the socket. */
@@ -80,7 +140,8 @@ int silc_socket_read(SilcSocketConnection sock)
       SILC_LOG_DEBUG(("Could not read immediately, will do it later"));
       return -2;
     }
-    SILC_LOG_ERROR(("Cannot read from socket: %d:%s", fd, strerror(errno)));
+    SILC_LOG_DEBUG(("Cannot read from socket: %d:%s", fd, strerror(errno)));
+    sock->sock_error = errno;
     return -1;
   }
 
@@ -101,5 +162,73 @@ int silc_socket_read(SilcSocketConnection sock)
 
   SILC_LOG_DEBUG(("Read %d bytes", len));
 
+  /* Apply QoS to the read data if necessary */
+  if (sock->qos) {
+    struct timeval curtime;
+    silc_gettimeofday(&curtime);
+
+    /* If we have passed the rate time limit, set our new time limit,
+       and zero the rate limit. */
+    if (!silc_compare_timeval(&curtime, &sock->qos->next_limit)) {
+      curtime.tv_sec++;
+      sock->qos->next_limit = curtime;
+      sock->qos->cur_rate = 0;
+    }
+    sock->qos->cur_rate++;
+
+    /* If we are not withing rate limit apply QoS for the read data */
+    if (sock->qos->cur_rate > sock->qos->read_rate) {
+      silc_schedule_task_add(sock->qos->schedule, sock->sock,
+                            silc_socket_read_qos, silc_socket_dup(sock),
+                            sock->qos->limit_sec, sock->qos->limit_usec,
+                            SILC_TASK_TIMEOUT, SILC_TASK_PRI_LOW);
+      silc_schedule_unset_listen_fd(sock->qos->schedule, sock->sock);
+
+      /* Check the byte limit as well, and do not return more than allowed */
+      if (sock->inbuf->len > sock->qos->read_limit_bytes) {
+       /* Hide the rest of the data from the buffer. */
+       sock->qos->data_len = sock->inbuf->len - sock->qos->read_limit_bytes;
+       silc_buffer_push_tail(sock->inbuf, sock->qos->data_len);
+       len = sock->inbuf->len;
+      } else {
+       /* Rate limit kicked in, do not return data yet */
+       return -2;
+      }
+    } else {
+      /* Check the byte limit, and do not return more than allowed */
+      if (sock->inbuf->len > sock->qos->read_limit_bytes) {
+       silc_schedule_task_add(sock->qos->schedule, sock->sock,
+                              silc_socket_read_qos, silc_socket_dup(sock),
+                              sock->qos->limit_sec, sock->qos->limit_usec,
+                              SILC_TASK_TIMEOUT, SILC_TASK_PRI_LOW);
+       silc_schedule_unset_listen_fd(sock->qos->schedule, sock->sock);
+
+       /* Hide the rest of the data from the buffer. */
+       sock->qos->data_len = sock->inbuf->len - sock->qos->read_limit_bytes;
+       silc_buffer_push_tail(sock->inbuf, sock->qos->data_len);
+       len = sock->inbuf->len;
+      }
+    }
+  }
+
   return len;
 }
+
+/* Returns human readable socket error message */
+
+bool silc_socket_get_error(SilcSocketConnection sock, char *error,
+                          SilcUInt32 error_len)
+{
+  char *err;
+
+  if (!sock->sock_error)
+    return FALSE;
+
+  err = strerror(sock->sock_error);
+  if (strlen(err) > error_len)
+    return FALSE;
+
+  memset(error, 0, error_len);
+  memcpy(error, err, strlen(err));
+  return TRUE;
+}