avf: handle flow id in data path 02/30002/22
authorChenmin Sun <chenmin.sun@intel.com>
Thu, 19 Nov 2020 21:06:41 +0000 (05:06 +0800)
committerDamjan Marion <dmarion@me.com>
Fri, 26 Mar 2021 08:30:54 +0000 (08:30 +0000)
Type: improvement

Signed-off-by: Chenmin Sun <chenmin.sun@intel.com>
Change-Id: I24fa52fee88dabad8f0b84446f796ee055819617

src/plugins/avf/avf.h
src/plugins/avf/format.c
src/plugins/avf/input.c

index 6c09084..51a81c6 100644 (file)
@@ -47,6 +47,7 @@
 #define AVF_RXD_STATUS(x)              (1ULL << x)
 #define AVF_RXD_STATUS_DD              AVF_RXD_STATUS(0)
 #define AVF_RXD_STATUS_EOP             AVF_RXD_STATUS(1)
+#define AVF_RXD_STATUS_FLM             AVF_RXD_STATUS (11)
 #define AVF_RXD_ERROR_SHIFT            19
 #define AVF_RXD_PTYPE_SHIFT            30
 #define AVF_RXD_LEN_SHIFT              38
@@ -134,6 +135,10 @@ typedef volatile struct
       u64 rsv2:3;
       u64 ptype:8;
       u64 length:26;
+
+      u64 rsv3 : 64;
+      u32 flex_lo;
+      u32 fdid_flex_hi;
     };
     u64 qword[4];
 #ifdef CLIB_HAVE_VEC256
@@ -298,7 +303,9 @@ typedef struct
 {
   CLIB_CACHE_LINE_ALIGN_MARK (cacheline0);
   vlib_buffer_t *bufs[AVF_RX_VECTOR_SZ];
+  u16 next[AVF_RX_VECTOR_SZ];
   u64 qw1s[AVF_RX_VECTOR_SZ];
+  u32 flow_ids[AVF_RX_VECTOR_SZ];
   avf_rx_tail_t tails[AVF_RX_VECTOR_SZ];
   vlib_buffer_t buffer_template;
 } avf_per_thread_data_t;
@@ -436,6 +443,7 @@ typedef struct
   u16 qid;
   u16 next_index;
   u32 hw_if_index;
+  u32 flow_id;
   u64 qw1s[AVF_RX_MAX_DESC_IN_CHAIN];
 } avf_input_trace_t;
 
index 2c4eaf7..9bdf4a5 100644 (file)
@@ -137,9 +137,9 @@ format_avf_input_trace (u8 * s, va_list * args)
   u32 indent = format_get_indent (s);
   int i = 0;
 
-  s = format (s, "avf: %v (%d) qid %u next-node %U",
-             hi->name, t->hw_if_index, t->qid, format_vlib_next_node_name,
-             vm, node->index, t->next_index);
+  s = format (s, "avf: %v (%d) qid %u next-node %U flow-id %u", hi->name,
+             t->hw_if_index, t->qid, format_vlib_next_node_name, vm,
+             node->index, t->next_index, t->flow_id);
 
   do
     {
index 221b54b..03ec288 100644 (file)
@@ -157,6 +157,37 @@ avf_rx_attach_tail (vlib_main_t * vm, vlib_buffer_t * bt, vlib_buffer_t * b,
   return tlnifb;
 }
 
+static_always_inline void
+avf_process_flow_offload (avf_device_t *ad, avf_per_thread_data_t *ptd,
+                         uword n_rx_packets)
+{
+  uword n;
+  avf_flow_lookup_entry_t *fle;
+
+  for (n = 0; n < n_rx_packets; n++)
+    {
+      if ((ptd->qw1s[n] & AVF_RXD_STATUS_FLM) == 0)
+       continue;
+
+      fle = pool_elt_at_index (ad->flow_lookup_entries, ptd->flow_ids[n]);
+
+      if (fle->next_index != (u16) ~0)
+       {
+         ptd->next[n] = fle->next_index;
+       }
+
+      if (fle->flow_id != ~0)
+       {
+         ptd->bufs[n]->flow_id = fle->flow_id;
+       }
+
+      if (fle->buffer_advance != ~0)
+       {
+         vlib_buffer_advance (ptd->bufs[n], fle->buffer_advance);
+       }
+    }
+}
+
 static_always_inline uword
 avf_process_rx_burst (vlib_main_t * vm, vlib_node_runtime_t * node,
                      avf_per_thread_data_t * ptd, u32 n_left,
@@ -210,6 +241,7 @@ avf_process_rx_burst (vlib_main_t * vm, vlib_node_runtime_t * node,
       b += 4;
       n_left -= 4;
     }
+
   while (n_left)
     {
       vlib_buffer_copy_template (b[0], &bt);
@@ -231,8 +263,9 @@ avf_process_rx_burst (vlib_main_t * vm, vlib_node_runtime_t * node,
 }
 
 static_always_inline uword
-avf_device_input_inline (vlib_main_t * vm, vlib_node_runtime_t * node,
-                        vlib_frame_t * frame, avf_device_t * ad, u16 qid)
+avf_device_input_inline (vlib_main_t *vm, vlib_node_runtime_t *node,
+                        vlib_frame_t *frame, avf_device_t *ad, u16 qid,
+                        int with_flows)
 {
   avf_main_t *am = &avf_main;
   vnet_main_t *vnm = vnet_get_main ();
@@ -252,11 +285,14 @@ avf_device_input_inline (vlib_main_t * vm, vlib_node_runtime_t * node,
   avf_rx_desc_t *d, *fd = rxq->descs;
 #ifdef CLIB_HAVE_VEC256
   u64x4 q1x4, or_q1x4 = { 0 };
+  u32x4 fdidx4;
   u64x4 dd_eop_mask4 = u64x4_splat (AVF_RXD_STATUS_DD | AVF_RXD_STATUS_EOP);
 #elif defined(CLIB_HAVE_VEC128)
   u32x4 q1x4_lo, q1x4_hi, or_q1x4 = { 0 };
+  u32x4 fdidx4;
   u32x4 dd_eop_mask4 = u32x4_splat (AVF_RXD_STATUS_DD | AVF_RXD_STATUS_EOP);
 #endif
+  int single_next = 1;
 
   /* is there anything on the ring */
   d = fd + next;
@@ -302,6 +338,7 @@ avf_device_input_inline (vlib_main_t * vm, vlib_node_runtime_t * node,
        goto one_by_one;
 
       or_q1x4 |= q1x4;
+
       u64x4_store_unaligned (q1x4, ptd->qw1s + n_rx_packets);
 #elif defined(CLIB_HAVE_VEC128)
       if (n_rx_packets >= AVF_RX_VECTOR_SZ - 4 || next >= size - 4)
@@ -326,6 +363,15 @@ avf_device_input_inline (vlib_main_t * vm, vlib_node_runtime_t * node,
       ptd->qw1s[n_rx_packets + 3] = (u64) q1x4_hi[3] << 32 | (u64) q1x4_lo[3];
 #endif
 #if defined(CLIB_HAVE_VEC256) || defined(CLIB_HAVE_VEC128)
+
+      if (with_flows)
+       {
+         fdidx4 = u32x4_gather (
+           (void *) &d[0].fdid_flex_hi, (void *) &d[1].fdid_flex_hi,
+           (void *) &d[2].fdid_flex_hi, (void *) &d[3].fdid_flex_hi);
+         u32x4_store_unaligned (fdidx4, ptd->flow_ids + n_rx_packets);
+       }
+
       vlib_buffer_copy_indices (bi, rxq->bufs + next, 4);
 
       /* next */
@@ -370,6 +416,10 @@ avf_device_input_inline (vlib_main_t * vm, vlib_node_runtime_t * node,
        }
 
       or_qw1 |= ptd->qw1s[n_rx_packets] = d[0].qword[1];
+      if (PREDICT_FALSE (with_flows))
+       {
+         ptd->flow_ids[n_rx_packets] = d[0].fdid_flex_hi;
+       }
 
       /* next */
       next = (next + 1) & mask;
@@ -401,15 +451,31 @@ no_more_desc:
   else
     n_rx_bytes = avf_process_rx_burst (vm, node, ptd, n_rx_packets, 0);
 
+  /* the MARKed packets may have different next nodes */
+  if (PREDICT_FALSE (with_flows && (or_qw1 & AVF_RXD_STATUS_FLM)))
+    {
+      u32 n;
+      single_next = 0;
+      for (n = 0; n < n_rx_packets; n++)
+       ptd->next[n] = next_index;
+
+      avf_process_flow_offload (ad, ptd, n_rx_packets);
+    }
+
   /* packet trace if enabled */
   if (PREDICT_FALSE ((n_trace = vlib_get_trace_count (vm, node))))
     {
-      u32 n_left = n_rx_packets, i = 0, j;
-      bi = to_next;
+      u32 n_left = n_rx_packets;
+      u32 i, j;
+      u16 *next_indices = ptd->next;
 
+      i = 0;
       while (n_trace && n_left)
        {
-         vlib_buffer_t *b = vlib_get_buffer (vm, bi[0]);
+         vlib_buffer_t *b = ptd->bufs[i];
+         if (PREDICT_FALSE (single_next == 0))
+           next_index = next_indices[0];
+
          if (PREDICT_TRUE
              (vlib_trace_buffer
               (vm, node, next_index, b, /* follow_chain */ 0)))
@@ -420,6 +486,8 @@ no_more_desc:
              tr->qid = qid;
              tr->hw_if_index = ad->hw_if_index;
              tr->qw1s[0] = ptd->qw1s[i];
+             tr->flow_id =
+               (tr->qw1s[0] & AVF_RXD_STATUS_FLM) ? ptd->flow_ids[i] : 0;
              for (j = 1; j < AVF_RX_MAX_DESC_IN_CHAIN; j++)
                tr->qw1s[j] = ptd->tails[i].qw1s[j - 1];
 
@@ -428,32 +496,46 @@ no_more_desc:
 
          /* next */
          n_left--;
-         bi++;
          i++;
+         next_indices++;
        }
       vlib_set_trace_count (vm, node, n_trace);
     }
 
-  if (PREDICT_TRUE (next_index == VNET_DEVICE_INPUT_NEXT_ETHERNET_INPUT))
+  /* enqueu the packets to the next nodes */
+  if (PREDICT_FALSE (with_flows && (or_qw1 & AVF_RXD_STATUS_FLM)))
     {
-      vlib_next_frame_t *nf;
-      vlib_frame_t *f;
-      ethernet_input_frame_t *ef;
-      nf = vlib_node_runtime_get_next_frame (vm, node, next_index);
-      f = vlib_get_frame (vm, nf->frame);
-      f->flags = ETH_INPUT_FRAME_F_SINGLE_SW_IF_IDX;
-
-      ef = vlib_frame_scalar_args (f);
-      ef->sw_if_index = ad->sw_if_index;
-      ef->hw_if_index = ad->hw_if_index;
-
-      if ((or_qw1 & AVF_RXD_ERROR_IPE) == 0)
-       f->flags |= ETH_INPUT_FRAME_F_IP4_CKSUM_OK;
-      vlib_frame_no_append (f);
+      /* release next node's frame vector, in this case we use
+        vlib_buffer_enqueue_to_next to place the packets
+       */
+      vlib_put_next_frame (vm, node, next_index, n_left_to_next);
+
+      /* enqueue buffers to the next node */
+      vlib_buffer_enqueue_to_next (vm, node, to_next, ptd->next, n_rx_packets);
     }
+  else
+    {
+      if (PREDICT_TRUE (next_index == VNET_DEVICE_INPUT_NEXT_ETHERNET_INPUT))
+       {
+         vlib_next_frame_t *nf;
+         vlib_frame_t *f;
+         ethernet_input_frame_t *ef;
+         nf = vlib_node_runtime_get_next_frame (vm, node, next_index);
+         f = vlib_get_frame (vm, nf->frame);
+         f->flags = ETH_INPUT_FRAME_F_SINGLE_SW_IF_IDX;
+
+         ef = vlib_frame_scalar_args (f);
+         ef->sw_if_index = ad->sw_if_index;
+         ef->hw_if_index = ad->hw_if_index;
+
+         if ((or_qw1 & AVF_RXD_ERROR_IPE) == 0)
+           f->flags |= ETH_INPUT_FRAME_F_IP4_CKSUM_OK;
+         vlib_frame_no_append (f);
+       }
 
-  n_left_to_next -= n_rx_packets;
-  vlib_put_next_frame (vm, node, next_index, n_left_to_next);
+      n_left_to_next -= n_rx_packets;
+      vlib_put_next_frame (vm, node, next_index, n_left_to_next);
+    }
 
   vlib_increment_combined_counter (vnm->interface_main.combined_sw_if_counters
                                   + VNET_INTERFACE_COUNTER_RX, thr_idx,
@@ -482,7 +564,12 @@ VLIB_NODE_FN (avf_input_node) (vlib_main_t * vm, vlib_node_runtime_t * node,
       avf_device_t *ad = avf_get_device (pv[i].dev_instance);
       if ((ad->flags & AVF_DEVICE_F_ADMIN_UP) == 0)
        continue;
-      n_rx += avf_device_input_inline (vm, node, frame, ad, pv[i].queue_id);
+      if (PREDICT_FALSE (ad->flags & AVF_DEVICE_F_RX_FLOW_OFFLOAD))
+       n_rx +=
+         avf_device_input_inline (vm, node, frame, ad, pv[i].queue_id, 1);
+      else
+       n_rx +=
+         avf_device_input_inline (vm, node, frame, ad, pv[i].queue_id, 0);
     }
 
   return n_rx;