aboutsummaryrefslogtreecommitdiffstats
path: root/drivers/infiniband/hw/nes
diff options
context:
space:
mode:
authorFaisal Latif <faisal.latif@intel.com>2008-11-21 21:50:49 -0500
committerRoland Dreier <rolandd@cisco.com>2008-12-05 14:00:13 -0500
commit4a14f6a79f5110c6033f0c61d77d07c449c2d083 (patch)
treefe582f37fae182a2bec420e7af8312dfa8d01acb /drivers/infiniband/hw/nes
parent183ecfa3091cd4cdda329a7fe89d9544088f517d (diff)
RDMA/nes: Forward packets for a new connection with stale APBVT entry
Under heavy traffic, there is a small windows when an APBVT entry is not yet removed and a new connection is established. Packets for the new connection are dropped until APBVT entry is removed. This patch will forward the packets instead of dropping them. Signed-off-by: Faisal Latif <faisal.latif@intel.com> Signed-off-by: Chien Tung <chien.tin.tung@intel.com> Signed-off-by: Roland Dreier <rolandd@cisco.com>
Diffstat (limited to 'drivers/infiniband/hw/nes')
-rw-r--r--drivers/infiniband/hw/nes/nes_cm.c44
-rw-r--r--drivers/infiniband/hw/nes/nes_cm.h2
-rw-r--r--drivers/infiniband/hw/nes/nes_hw.c42
3 files changed, 45 insertions, 43 deletions
diff --git a/drivers/infiniband/hw/nes/nes_cm.c b/drivers/infiniband/hw/nes/nes_cm.c
index c259ddc8dd88..f34fa6f77a81 100644
--- a/drivers/infiniband/hw/nes/nes_cm.c
+++ b/drivers/infiniband/hw/nes/nes_cm.c
@@ -86,7 +86,7 @@ static int mini_cm_accept(struct nes_cm_core *, struct ietf_mpa_frame *,
86 struct nes_cm_node *); 86 struct nes_cm_node *);
87static int mini_cm_reject(struct nes_cm_core *, struct ietf_mpa_frame *, 87static int mini_cm_reject(struct nes_cm_core *, struct ietf_mpa_frame *,
88 struct nes_cm_node *); 88 struct nes_cm_node *);
89static void mini_cm_recv_pkt(struct nes_cm_core *, struct nes_vnic *, 89static int mini_cm_recv_pkt(struct nes_cm_core *, struct nes_vnic *,
90 struct sk_buff *); 90 struct sk_buff *);
91static int mini_cm_dealloc_core(struct nes_cm_core *); 91static int mini_cm_dealloc_core(struct nes_cm_core *);
92static int mini_cm_get(struct nes_cm_core *); 92static int mini_cm_get(struct nes_cm_core *);
@@ -2076,7 +2076,7 @@ static int mini_cm_close(struct nes_cm_core *cm_core, struct nes_cm_node *cm_nod
2076 * recv_pkt - recv an ETHERNET packet, and process it through CM 2076 * recv_pkt - recv an ETHERNET packet, and process it through CM
2077 * node state machine 2077 * node state machine
2078 */ 2078 */
2079static void mini_cm_recv_pkt(struct nes_cm_core *cm_core, 2079static int mini_cm_recv_pkt(struct nes_cm_core *cm_core,
2080 struct nes_vnic *nesvnic, struct sk_buff *skb) 2080 struct nes_vnic *nesvnic, struct sk_buff *skb)
2081{ 2081{
2082 struct nes_cm_node *cm_node = NULL; 2082 struct nes_cm_node *cm_node = NULL;
@@ -2084,23 +2084,16 @@ static void mini_cm_recv_pkt(struct nes_cm_core *cm_core,
2084 struct iphdr *iph; 2084 struct iphdr *iph;
2085 struct tcphdr *tcph; 2085 struct tcphdr *tcph;
2086 struct nes_cm_info nfo; 2086 struct nes_cm_info nfo;
2087 int skb_handled = 1;
2087 2088
2088 if (!skb) 2089 if (!skb)
2089 return; 2090 return 0;
2090 if (skb->len < sizeof(struct iphdr) + sizeof(struct tcphdr)) { 2091 if (skb->len < sizeof(struct iphdr) + sizeof(struct tcphdr)) {
2091 dev_kfree_skb_any(skb); 2092 return 0;
2092 return;
2093 } 2093 }
2094 2094
2095 iph = (struct iphdr *)skb->data; 2095 iph = (struct iphdr *)skb->data;
2096 tcph = (struct tcphdr *)(skb->data + sizeof(struct iphdr)); 2096 tcph = (struct tcphdr *)(skb->data + sizeof(struct iphdr));
2097 skb_reset_network_header(skb);
2098 skb_set_transport_header(skb, sizeof(*tcph));
2099 if (!tcph) {
2100 dev_kfree_skb_any(skb);
2101 return;
2102 }
2103 skb->len = ntohs(iph->tot_len);
2104 2097
2105 nfo.loc_addr = ntohl(iph->daddr); 2098 nfo.loc_addr = ntohl(iph->daddr);
2106 nfo.loc_port = ntohs(tcph->dest); 2099 nfo.loc_port = ntohs(tcph->dest);
@@ -2121,23 +2114,21 @@ static void mini_cm_recv_pkt(struct nes_cm_core *cm_core,
2121 /* Only type of packet accepted are for */ 2114 /* Only type of packet accepted are for */
2122 /* the PASSIVE open (syn only) */ 2115 /* the PASSIVE open (syn only) */
2123 if ((!tcph->syn) || (tcph->ack)) { 2116 if ((!tcph->syn) || (tcph->ack)) {
2124 cm_packets_dropped++; 2117 skb_handled = 0;
2125 break; 2118 break;
2126 } 2119 }
2127 listener = find_listener(cm_core, nfo.loc_addr, 2120 listener = find_listener(cm_core, nfo.loc_addr,
2128 nfo.loc_port, 2121 nfo.loc_port,
2129 NES_CM_LISTENER_ACTIVE_STATE); 2122 NES_CM_LISTENER_ACTIVE_STATE);
2130 if (listener) { 2123 if (!listener) {
2131 nfo.cm_id = listener->cm_id; 2124 nfo.cm_id = NULL;
2132 nfo.conn_type = listener->conn_type; 2125 nfo.conn_type = 0;
2133 } else { 2126 nes_debug(NES_DBG_CM, "Unable to find listener for the pkt\n");
2134 nes_debug(NES_DBG_CM, "Unable to find listener " 2127 skb_handled = 0;
2135 "for the pkt\n");
2136 cm_packets_dropped++;
2137 dev_kfree_skb_any(skb);
2138 break; 2128 break;
2139 } 2129 }
2140 2130 nfo.cm_id = listener->cm_id;
2131 nfo.conn_type = listener->conn_type;
2141 cm_node = make_cm_node(cm_core, nesvnic, &nfo, 2132 cm_node = make_cm_node(cm_core, nesvnic, &nfo,
2142 listener); 2133 listener);
2143 if (!cm_node) { 2134 if (!cm_node) {
@@ -2163,9 +2154,13 @@ static void mini_cm_recv_pkt(struct nes_cm_core *cm_core,
2163 dev_kfree_skb_any(skb); 2154 dev_kfree_skb_any(skb);
2164 break; 2155 break;
2165 } 2156 }
2157 skb_reset_network_header(skb);
2158 skb_set_transport_header(skb, sizeof(*tcph));
2159 skb->len = ntohs(iph->tot_len);
2166 process_packet(cm_node, skb, cm_core); 2160 process_packet(cm_node, skb, cm_core);
2167 rem_ref_cm_node(cm_core, cm_node); 2161 rem_ref_cm_node(cm_core, cm_node);
2168 } while (0); 2162 } while (0);
2163 return skb_handled;
2169} 2164}
2170 2165
2171 2166
@@ -2985,15 +2980,16 @@ int nes_destroy_listen(struct iw_cm_id *cm_id)
2985 */ 2980 */
2986int nes_cm_recv(struct sk_buff *skb, struct net_device *netdevice) 2981int nes_cm_recv(struct sk_buff *skb, struct net_device *netdevice)
2987{ 2982{
2983 int rc = 0;
2988 cm_packets_received++; 2984 cm_packets_received++;
2989 if ((g_cm_core) && (g_cm_core->api)) { 2985 if ((g_cm_core) && (g_cm_core->api)) {
2990 g_cm_core->api->recv_pkt(g_cm_core, netdev_priv(netdevice), skb); 2986 rc = g_cm_core->api->recv_pkt(g_cm_core, netdev_priv(netdevice), skb);
2991 } else { 2987 } else {
2992 nes_debug(NES_DBG_CM, "Unable to process packet for CM," 2988 nes_debug(NES_DBG_CM, "Unable to process packet for CM,"
2993 " cm is not setup properly.\n"); 2989 " cm is not setup properly.\n");
2994 } 2990 }
2995 2991
2996 return 0; 2992 return rc;
2997} 2993}
2998 2994
2999 2995
diff --git a/drivers/infiniband/hw/nes/nes_cm.h b/drivers/infiniband/hw/nes/nes_cm.h
index 25e2493abc6a..3a20a7883976 100644
--- a/drivers/infiniband/hw/nes/nes_cm.h
+++ b/drivers/infiniband/hw/nes/nes_cm.h
@@ -397,7 +397,7 @@ struct nes_cm_ops {
397 struct nes_cm_node *); 397 struct nes_cm_node *);
398 int (*reject)(struct nes_cm_core *, struct ietf_mpa_frame *, 398 int (*reject)(struct nes_cm_core *, struct ietf_mpa_frame *,
399 struct nes_cm_node *); 399 struct nes_cm_node *);
400 void (*recv_pkt)(struct nes_cm_core *, struct nes_vnic *, 400 int (*recv_pkt)(struct nes_cm_core *, struct nes_vnic *,
401 struct sk_buff *); 401 struct sk_buff *);
402 int (*destroy_cm_core)(struct nes_cm_core *); 402 int (*destroy_cm_core)(struct nes_cm_core *);
403 int (*get)(struct nes_cm_core *); 403 int (*get)(struct nes_cm_core *);
diff --git a/drivers/infiniband/hw/nes/nes_hw.c b/drivers/infiniband/hw/nes/nes_hw.c
index 7c49cc882d75..8f70ff2dcc58 100644
--- a/drivers/infiniband/hw/nes/nes_hw.c
+++ b/drivers/infiniband/hw/nes/nes_hw.c
@@ -2700,27 +2700,33 @@ void nes_nic_ce_handler(struct nes_device *nesdev, struct nes_hw_nic_cq *cq)
2700 pkt_type, (pkt_type & NES_PKT_TYPE_APBVT_MASK)); */ 2700 pkt_type, (pkt_type & NES_PKT_TYPE_APBVT_MASK)); */
2701 2701
2702 if ((pkt_type & NES_PKT_TYPE_APBVT_MASK) == NES_PKT_TYPE_APBVT_BITS) { 2702 if ((pkt_type & NES_PKT_TYPE_APBVT_MASK) == NES_PKT_TYPE_APBVT_BITS) {
2703 nes_cm_recv(rx_skb, nesvnic->netdev); 2703 if (nes_cm_recv(rx_skb, nesvnic->netdev))
2704 rx_skb = NULL;
2705 }
2706 if (rx_skb == NULL)
2707 goto skip_rx_indicate0;
2708
2709
2710 if ((cqe_misc & NES_NIC_CQE_TAG_VALID) &&
2711 (nesvnic->vlan_grp != NULL)) {
2712 vlan_tag = (u16)(le32_to_cpu(
2713 cq->cq_vbase[head].cqe_words[NES_NIC_CQE_TAG_PKT_TYPE_IDX])
2714 >> 16);
2715 nes_debug(NES_DBG_CQ, "%s: Reporting stripped VLAN packet. Tag = 0x%04X\n",
2716 nesvnic->netdev->name, vlan_tag);
2717 if (nes_use_lro)
2718 lro_vlan_hwaccel_receive_skb(&nesvnic->lro_mgr, rx_skb,
2719 nesvnic->vlan_grp, vlan_tag, NULL);
2720 else
2721 nes_vlan_rx(rx_skb, nesvnic->vlan_grp, vlan_tag);
2704 } else { 2722 } else {
2705 if ((cqe_misc & NES_NIC_CQE_TAG_VALID) && (nesvnic->vlan_grp != NULL)) { 2723 if (nes_use_lro)
2706 vlan_tag = (u16)(le32_to_cpu( 2724 lro_receive_skb(&nesvnic->lro_mgr, rx_skb, NULL);
2707 cq->cq_vbase[head].cqe_words[NES_NIC_CQE_TAG_PKT_TYPE_IDX]) 2725 else
2708 >> 16); 2726 nes_netif_rx(rx_skb);
2709 nes_debug(NES_DBG_CQ, "%s: Reporting stripped VLAN packet. Tag = 0x%04X\n",
2710 nesvnic->netdev->name, vlan_tag);
2711 if (nes_use_lro)
2712 lro_vlan_hwaccel_receive_skb(&nesvnic->lro_mgr, rx_skb,
2713 nesvnic->vlan_grp, vlan_tag, NULL);
2714 else
2715 nes_vlan_rx(rx_skb, nesvnic->vlan_grp, vlan_tag);
2716 } else {
2717 if (nes_use_lro)
2718 lro_receive_skb(&nesvnic->lro_mgr, rx_skb, NULL);
2719 else
2720 nes_netif_rx(rx_skb);
2721 }
2722 } 2727 }
2723 2728
2729skip_rx_indicate0:
2724 nesvnic->netdev->last_rx = jiffies; 2730 nesvnic->netdev->last_rx = jiffies;
2725 /* nesvnic->netstats.rx_packets++; */ 2731 /* nesvnic->netstats.rx_packets++; */
2726 /* nesvnic->netstats.rx_bytes += rx_pkt_size; */ 2732 /* nesvnic->netstats.rx_bytes += rx_pkt_size; */