crypto: chtls - Inline TLS record Rx
authorAtul Gupta <atul.gupta@chelsio.com>
Sat, 31 Mar 2018 16:12:01 +0000 (21:42 +0530)
committerDavid S. Miller <davem@davemloft.net>
Sun, 1 Apr 2018 03:37:33 +0000 (23:37 -0400)
handler for record receive. plain text copied to user
buffer

Signed-off-by: Atul Gupta <atul.gupta@chelsio.com>
Signed-off-by: Michael Werner <werner@chelsio.com>
Signed-off-by: David S. Miller <davem@davemloft.net>
drivers/crypto/chelsio/chtls/chtls_io.c
drivers/crypto/chelsio/chtls/chtls_main.c

index 6974d3ed0ff52111dc6026c87b60bec1cdcdd192..5a75be43950f6bebe244c5b30bc4f3e2cdf44d40 100644 (file)
@@ -30,6 +30,11 @@ static bool is_tls_tx(struct chtls_sock *csk)
        return csk->tlshws.txkey >= 0;
 }
 
+static bool is_tls_rx(struct chtls_sock *csk)
+{
+       return csk->tlshws.rxkey >= 0;
+}
+
 static int data_sgl_len(const struct sk_buff *skb)
 {
        unsigned int cnt;
@@ -106,10 +111,12 @@ static int send_flowc_wr(struct sock *sk, struct fw_flowc_wr *flowc,
 {
        struct chtls_sock *csk = rcu_dereference_sk_user_data(sk);
        struct tcp_sock *tp = tcp_sk(sk);
-       int flowclen16 = flowclen / 16;
        struct sk_buff *skb;
+       int flowclen16;
        int ret;
 
+       flowclen16 = flowclen / 16;
+
        if (csk_flag(sk, CSK_TX_DATA_SENT)) {
                skb = create_flowc_wr_skb(sk, flowc, flowclen);
                if (!skb)
@@ -1220,3 +1227,596 @@ out_err:
        copied = sk_stream_error(sk, flags, err);
        goto done;
 }
+
+static void chtls_select_window(struct sock *sk)
+{
+       struct chtls_sock *csk = rcu_dereference_sk_user_data(sk);
+       struct tcp_sock *tp = tcp_sk(sk);
+       unsigned int wnd = tp->rcv_wnd;
+
+       wnd = max_t(unsigned int, wnd, tcp_full_space(sk));
+       wnd = max_t(unsigned int, MIN_RCV_WND, wnd);
+
+       if (wnd > MAX_RCV_WND)
+               wnd = MAX_RCV_WND;
+
+/*
+ * Check if we need to grow the receive window in response to an increase in
+ * the socket's receive buffer size.  Some applications increase the buffer
+ * size dynamically and rely on the window to grow accordingly.
+ */
+
+       if (wnd > tp->rcv_wnd) {
+               tp->rcv_wup -= wnd - tp->rcv_wnd;
+               tp->rcv_wnd = wnd;
+               /* Mark the receive window as updated */
+               csk_reset_flag(csk, CSK_UPDATE_RCV_WND);
+       }
+}
+
+/*
+ * Send RX credits through an RX_DATA_ACK CPL message.  We are permitted
+ * to return without sending the message in case we cannot allocate
+ * an sk_buff.  Returns the number of credits sent.
+ */
+static u32 send_rx_credits(struct chtls_sock *csk, u32 credits)
+{
+       struct cpl_rx_data_ack *req;
+       struct sk_buff *skb;
+
+       skb = alloc_skb(sizeof(*req), GFP_ATOMIC);
+       if (!skb)
+               return 0;
+       __skb_put(skb, sizeof(*req));
+       req = (struct cpl_rx_data_ack *)skb->head;
+
+       set_wr_txq(skb, CPL_PRIORITY_ACK, csk->port_id);
+       INIT_TP_WR(req, csk->tid);
+       OPCODE_TID(req) = cpu_to_be32(MK_OPCODE_TID(CPL_RX_DATA_ACK,
+                                                   csk->tid));
+       req->credit_dack = cpu_to_be32(RX_CREDITS_V(credits) |
+                                      RX_FORCE_ACK_F);
+       cxgb4_ofld_send(csk->cdev->ports[csk->port_id], skb);
+       return credits;
+}
+
+#define CREDIT_RETURN_STATE (TCPF_ESTABLISHED | \
+                            TCPF_FIN_WAIT1 | \
+                            TCPF_FIN_WAIT2)
+
+/*
+ * Called after some received data has been read.  It returns RX credits
+ * to the HW for the amount of data processed.
+ */
+static void chtls_cleanup_rbuf(struct sock *sk, int copied)
+{
+       struct chtls_sock *csk = rcu_dereference_sk_user_data(sk);
+       struct tcp_sock *tp;
+       int must_send;
+       u32 credits;
+       u32 thres;
+
+       thres = 15 * 1024;
+
+       if (!sk_in_state(sk, CREDIT_RETURN_STATE))
+               return;
+
+       chtls_select_window(sk);
+       tp = tcp_sk(sk);
+       credits = tp->copied_seq - tp->rcv_wup;
+       if (unlikely(!credits))
+               return;
+
+/*
+ * For coalescing to work effectively ensure the receive window has
+ * at least 16KB left.
+ */
+       must_send = credits + 16384 >= tp->rcv_wnd;
+
+       if (must_send || credits >= thres)
+               tp->rcv_wup += send_rx_credits(csk, credits);
+}
+
+static int chtls_pt_recvmsg(struct sock *sk, struct msghdr *msg, size_t len,
+                           int nonblock, int flags, int *addr_len)
+{
+       struct chtls_sock *csk = rcu_dereference_sk_user_data(sk);
+       struct net_device *dev = csk->egress_dev;
+       struct chtls_hws *hws = &csk->tlshws;
+       struct tcp_sock *tp = tcp_sk(sk);
+       struct adapter *adap;
+       unsigned long avail;
+       int buffers_freed;
+       int copied = 0;
+       int request;
+       int target;
+       long timeo;
+
+       adap = netdev2adap(dev);
+       buffers_freed = 0;
+
+       timeo = sock_rcvtimeo(sk, nonblock);
+       target = sock_rcvlowat(sk, flags & MSG_WAITALL, len);
+       request = len;
+
+       if (unlikely(csk_flag(sk, CSK_UPDATE_RCV_WND)))
+               chtls_cleanup_rbuf(sk, copied);
+
+       do {
+               struct sk_buff *skb;
+               u32 offset = 0;
+
+               if (unlikely(tp->urg_data &&
+                            tp->urg_seq == tp->copied_seq)) {
+                       if (copied)
+                               break;
+                       if (signal_pending(current)) {
+                               copied = timeo ? sock_intr_errno(timeo) :
+                                       -EAGAIN;
+                               break;
+                       }
+               }
+               skb = skb_peek(&sk->sk_receive_queue);
+               if (skb)
+                       goto found_ok_skb;
+               if (csk->wr_credits &&
+                   skb_queue_len(&csk->txq) &&
+                   chtls_push_frames(csk, csk->wr_credits ==
+                                     csk->wr_max_credits))
+                       sk->sk_write_space(sk);
+
+               if (copied >= target && !sk->sk_backlog.tail)
+                       break;
+
+               if (copied) {
+                       if (sk->sk_err || sk->sk_state == TCP_CLOSE ||
+                           (sk->sk_shutdown & RCV_SHUTDOWN) ||
+                           signal_pending(current))
+                               break;
+
+                       if (!timeo)
+                               break;
+               } else {
+                       if (sock_flag(sk, SOCK_DONE))
+                               break;
+                       if (sk->sk_err) {
+                               copied = sock_error(sk);
+                               break;
+                       }
+                       if (sk->sk_shutdown & RCV_SHUTDOWN)
+                               break;
+                       if (sk->sk_state == TCP_CLOSE) {
+                               copied = -ENOTCONN;
+                               break;
+                       }
+                       if (!timeo) {
+                               copied = -EAGAIN;
+                               break;
+                       }
+                       if (signal_pending(current)) {
+                               copied = sock_intr_errno(timeo);
+                               break;
+                       }
+               }
+               if (sk->sk_backlog.tail) {
+                       release_sock(sk);
+                       lock_sock(sk);
+                       chtls_cleanup_rbuf(sk, copied);
+                       continue;
+               }
+
+               if (copied >= target)
+                       break;
+               chtls_cleanup_rbuf(sk, copied);
+               sk_wait_data(sk, &timeo, NULL);
+                       continue;
+found_ok_skb:
+               if (!skb->len) {
+                       skb_dst_set(skb, NULL);
+                       __skb_unlink(skb, &sk->sk_receive_queue);
+                       kfree_skb(skb);
+
+                       if (!copied && !timeo) {
+                               copied = -EAGAIN;
+                               break;
+                       }
+
+                       if (copied < target) {
+                               release_sock(sk);
+                               lock_sock(sk);
+                               continue;
+                       }
+                       break;
+               }
+               offset = hws->copied_seq;
+               avail = skb->len - offset;
+               if (len < avail)
+                       avail = len;
+
+               if (unlikely(tp->urg_data)) {
+                       u32 urg_offset = tp->urg_seq - tp->copied_seq;
+
+                       if (urg_offset < avail) {
+                               if (urg_offset) {
+                                       avail = urg_offset;
+                               } else if (!sock_flag(sk, SOCK_URGINLINE)) {
+                                       /* First byte is urgent, skip */
+                                       tp->copied_seq++;
+                                       offset++;
+                                       avail--;
+                                       if (!avail)
+                                               goto skip_copy;
+                               }
+                       }
+               }
+               if (hws->rstate == TLS_RCV_ST_READ_BODY) {
+                       if (skb_copy_datagram_msg(skb, offset,
+                                                 msg, avail)) {
+                               if (!copied) {
+                                       copied = -EFAULT;
+                                       break;
+                               }
+                       }
+               } else {
+                       struct tlsrx_cmp_hdr *tls_hdr_pkt =
+                               (struct tlsrx_cmp_hdr *)skb->data;
+
+                       if ((tls_hdr_pkt->res_to_mac_error &
+                           TLSRX_HDR_PKT_ERROR_M))
+                               tls_hdr_pkt->type = 0x7F;
+
+                       /* CMP pld len is for recv seq */
+                       hws->rcvpld = skb->hdr_len;
+                       if (skb_copy_datagram_msg(skb, offset, msg, avail)) {
+                               if (!copied) {
+                                       copied = -EFAULT;
+                                       break;
+                               }
+                       }
+               }
+               copied += avail;
+               len -= avail;
+               hws->copied_seq += avail;
+skip_copy:
+               if (tp->urg_data && after(tp->copied_seq, tp->urg_seq))
+                       tp->urg_data = 0;
+
+               if (hws->rstate == TLS_RCV_ST_READ_BODY &&
+                   (avail + offset) >= skb->len) {
+                       if (likely(skb))
+                               chtls_free_skb(sk, skb);
+                       buffers_freed++;
+                       hws->rstate = TLS_RCV_ST_READ_HEADER;
+                       atomic_inc(&adap->chcr_stats.tls_pdu_rx);
+                       tp->copied_seq += hws->rcvpld;
+                       hws->copied_seq = 0;
+                       if (copied >= target &&
+                           !skb_peek(&sk->sk_receive_queue))
+                               break;
+               } else {
+                       if (likely(skb)) {
+                               if (ULP_SKB_CB(skb)->flags &
+                                   ULPCB_FLAG_TLS_ND)
+                                       hws->rstate =
+                                               TLS_RCV_ST_READ_HEADER;
+                               else
+                                       hws->rstate =
+                                               TLS_RCV_ST_READ_BODY;
+                               chtls_free_skb(sk, skb);
+                       }
+                       buffers_freed++;
+                       tp->copied_seq += avail;
+                       hws->copied_seq = 0;
+               }
+       } while (len > 0);
+
+       if (buffers_freed)
+               chtls_cleanup_rbuf(sk, copied);
+       release_sock(sk);
+       return copied;
+}
+
+/*
+ * Peek at data in a socket's receive buffer.
+ */
+static int peekmsg(struct sock *sk, struct msghdr *msg,
+                  size_t len, int nonblock, int flags)
+{
+       struct tcp_sock *tp = tcp_sk(sk);
+       u32 peek_seq, offset;
+       struct sk_buff *skb;
+       int copied = 0;
+       size_t avail;          /* amount of available data in current skb */
+       long timeo;
+
+       lock_sock(sk);
+       timeo = sock_rcvtimeo(sk, nonblock);
+       peek_seq = tp->copied_seq;
+
+       do {
+               if (unlikely(tp->urg_data && tp->urg_seq == peek_seq)) {
+                       if (copied)
+                               break;
+                       if (signal_pending(current)) {
+                               copied = timeo ? sock_intr_errno(timeo) :
+                               -EAGAIN;
+                               break;
+                       }
+               }
+
+               skb_queue_walk(&sk->sk_receive_queue, skb) {
+                       offset = peek_seq - ULP_SKB_CB(skb)->seq;
+                       if (offset < skb->len)
+                               goto found_ok_skb;
+               }
+
+               /* empty receive queue */
+               if (copied)
+                       break;
+               if (sock_flag(sk, SOCK_DONE))
+                       break;
+               if (sk->sk_err) {
+                       copied = sock_error(sk);
+                       break;
+               }
+               if (sk->sk_shutdown & RCV_SHUTDOWN)
+                       break;
+               if (sk->sk_state == TCP_CLOSE) {
+                       copied = -ENOTCONN;
+                       break;
+               }
+               if (!timeo) {
+                       copied = -EAGAIN;
+                       break;
+               }
+               if (signal_pending(current)) {
+                       copied = sock_intr_errno(timeo);
+                       break;
+               }
+
+               if (sk->sk_backlog.tail) {
+                       /* Do not sleep, just process backlog. */
+                       release_sock(sk);
+                       lock_sock(sk);
+               } else {
+                       sk_wait_data(sk, &timeo, NULL);
+               }
+
+               if (unlikely(peek_seq != tp->copied_seq)) {
+                       if (net_ratelimit())
+                               pr_info("TCP(%s:%d), race in MSG_PEEK.\n",
+                                       current->comm, current->pid);
+                       peek_seq = tp->copied_seq;
+               }
+               continue;
+
+found_ok_skb:
+               avail = skb->len - offset;
+               if (len < avail)
+                       avail = len;
+               /*
+                * Do we have urgent data here?  We need to skip over the
+                * urgent byte.
+                */
+               if (unlikely(tp->urg_data)) {
+                       u32 urg_offset = tp->urg_seq - peek_seq;
+
+                       if (urg_offset < avail) {
+                               /*
+                                * The amount of data we are preparing to copy
+                                * contains urgent data.
+                                */
+                               if (!urg_offset) { /* First byte is urgent */
+                                       if (!sock_flag(sk, SOCK_URGINLINE)) {
+                                               peek_seq++;
+                                               offset++;
+                                               avail--;
+                                       }
+                                       if (!avail)
+                                               continue;
+                               } else {
+                                       /* stop short of the urgent data */
+                                       avail = urg_offset;
+                               }
+                       }
+               }
+
+               /*
+                * If MSG_TRUNC is specified the data is discarded.
+                */
+               if (likely(!(flags & MSG_TRUNC)))
+                       if (skb_copy_datagram_msg(skb, offset, msg, len)) {
+                               if (!copied) {
+                                       copied = -EFAULT;
+                                       break;
+                               }
+                       }
+               peek_seq += avail;
+               copied += avail;
+               len -= avail;
+       } while (len > 0);
+
+       release_sock(sk);
+       return copied;
+}
+
+int chtls_recvmsg(struct sock *sk, struct msghdr *msg, size_t len,
+                 int nonblock, int flags, int *addr_len)
+{
+       struct tcp_sock *tp = tcp_sk(sk);
+       struct chtls_sock *csk;
+       struct chtls_hws *hws;
+       unsigned long avail;    /* amount of available data in current skb */
+       int buffers_freed;
+       int copied = 0;
+       int request;
+       long timeo;
+       int target;             /* Read at least this many bytes */
+
+       buffers_freed = 0;
+
+       if (unlikely(flags & MSG_OOB))
+               return tcp_prot.recvmsg(sk, msg, len, nonblock, flags,
+                                       addr_len);
+
+       if (unlikely(flags & MSG_PEEK))
+               return peekmsg(sk, msg, len, nonblock, flags);
+
+       if (sk_can_busy_loop(sk) &&
+           skb_queue_empty(&sk->sk_receive_queue) &&
+           sk->sk_state == TCP_ESTABLISHED)
+               sk_busy_loop(sk, nonblock);
+
+       lock_sock(sk);
+       csk = rcu_dereference_sk_user_data(sk);
+       hws = &csk->tlshws;
+
+       if (is_tls_rx(csk))
+               return chtls_pt_recvmsg(sk, msg, len, nonblock,
+                                       flags, addr_len);
+
+       timeo = sock_rcvtimeo(sk, nonblock);
+       target = sock_rcvlowat(sk, flags & MSG_WAITALL, len);
+       request = len;
+
+       if (unlikely(csk_flag(sk, CSK_UPDATE_RCV_WND)))
+               chtls_cleanup_rbuf(sk, copied);
+
+       do {
+               struct sk_buff *skb;
+               u32 offset;
+
+               if (unlikely(tp->urg_data && tp->urg_seq == tp->copied_seq)) {
+                       if (copied)
+                               break;
+                       if (signal_pending(current)) {
+                               copied = timeo ? sock_intr_errno(timeo) :
+                                       -EAGAIN;
+                               break;
+                       }
+               }
+
+               skb = skb_peek(&sk->sk_receive_queue);
+               if (skb)
+                       goto found_ok_skb;
+
+               if (csk->wr_credits &&
+                   skb_queue_len(&csk->txq) &&
+                   chtls_push_frames(csk, csk->wr_credits ==
+                                     csk->wr_max_credits))
+                       sk->sk_write_space(sk);
+
+               if (copied >= target && !sk->sk_backlog.tail)
+                       break;
+
+               if (copied) {
+                       if (sk->sk_err || sk->sk_state == TCP_CLOSE ||
+                           (sk->sk_shutdown & RCV_SHUTDOWN) ||
+                           signal_pending(current))
+                               break;
+               } else {
+                       if (sock_flag(sk, SOCK_DONE))
+                               break;
+                       if (sk->sk_err) {
+                               copied = sock_error(sk);
+                               break;
+                       }
+                       if (sk->sk_shutdown & RCV_SHUTDOWN)
+                               break;
+                       if (sk->sk_state == TCP_CLOSE) {
+                               copied = -ENOTCONN;
+                               break;
+                       }
+                       if (!timeo) {
+                               copied = -EAGAIN;
+                               break;
+                       }
+                       if (signal_pending(current)) {
+                               copied = sock_intr_errno(timeo);
+                               break;
+                       }
+               }
+
+               if (sk->sk_backlog.tail) {
+                       release_sock(sk);
+                       lock_sock(sk);
+                       chtls_cleanup_rbuf(sk, copied);
+                       continue;
+               }
+
+               if (copied >= target)
+                       break;
+               chtls_cleanup_rbuf(sk, copied);
+               sk_wait_data(sk, &timeo, NULL);
+               continue;
+
+found_ok_skb:
+               if (!skb->len) {
+                       chtls_kfree_skb(sk, skb);
+                       if (!copied && !timeo) {
+                               copied = -EAGAIN;
+                               break;
+                       }
+
+                       if (copied < target)
+                               continue;
+
+                       break;
+               }
+
+               offset = tp->copied_seq - ULP_SKB_CB(skb)->seq;
+               avail = skb->len - offset;
+               if (len < avail)
+                       avail = len;
+
+               if (unlikely(tp->urg_data)) {
+                       u32 urg_offset = tp->urg_seq - tp->copied_seq;
+
+                       if (urg_offset < avail) {
+                               if (urg_offset) {
+                                       avail = urg_offset;
+                               } else if (!sock_flag(sk, SOCK_URGINLINE)) {
+                                       tp->copied_seq++;
+                                       offset++;
+                                       avail--;
+                                       if (!avail)
+                                               goto skip_copy;
+                               }
+                       }
+               }
+
+               if (likely(!(flags & MSG_TRUNC))) {
+                       if (skb_copy_datagram_msg(skb, offset,
+                                                 msg, avail)) {
+                               if (!copied) {
+                                       copied = -EFAULT;
+                                       break;
+                               }
+                       }
+               }
+
+               tp->copied_seq += avail;
+               copied += avail;
+               len -= avail;
+
+skip_copy:
+               if (tp->urg_data && after(tp->copied_seq, tp->urg_seq))
+                       tp->urg_data = 0;
+
+               if (avail + offset >= skb->len) {
+                       if (likely(skb))
+                               chtls_free_skb(sk, skb);
+                       buffers_freed++;
+
+                       if  (copied >= target &&
+                            !skb_peek(&sk->sk_receive_queue))
+                               break;
+               }
+       } while (len > 0);
+
+       if (buffers_freed)
+               chtls_cleanup_rbuf(sk, copied);
+
+       release_sock(sk);
+       return copied;
+}
index e5e543a455428d3853c7fbf82296947d021ec550..007c45c38fc71e8dca4eb7d77cc8b37cc3056f8f 100644 (file)
@@ -549,6 +549,7 @@ static void __init chtls_init_ulp_ops(void)
        chtls_cpl_prot.shutdown         = chtls_shutdown;
        chtls_cpl_prot.sendmsg          = chtls_sendmsg;
        chtls_cpl_prot.sendpage         = chtls_sendpage;
+       chtls_cpl_prot.recvmsg          = chtls_recvmsg;
        chtls_cpl_prot.setsockopt       = chtls_setsockopt;
        chtls_cpl_prot.getsockopt       = chtls_getsockopt;
 }