[v2] net/softnic: add support for flow API vxlan encap action

Message ID 1539614504-29090-1-git-send-email-reshma.pattan@intel.com (mailing list archive)
State Superseded, archived
Delegated to: Cristian Dumitrescu
Headers
Series [v2] net/softnic: add support for flow API vxlan encap action |

Checks

Context Check Description
ci/Intel-compilation success Compilation OK
ci/checkpatch success coding style OK

Commit Message

Pattan, Reshma Oct. 15, 2018, 2:41 p.m. UTC
Added support to handle vxlan encap action of rte flow rule.

Signed-off-by: Reshma Pattan <reshma.pattan@intel.com>
Acked-by: Dumitrescu Cristian <cristian.dumitrescu@intel.com>
---
 drivers/net/softnic/rte_eth_softnic_flow.c | 192 +++++++++++++++++++++
 1 file changed, 192 insertions(+)
  

Comments

Cristian Dumitrescu Oct. 15, 2018, 4:05 p.m. UTC | #1
> -----Original Message-----
> From: Pattan, Reshma
> Sent: Monday, October 15, 2018 3:42 PM
> To: dev@dpdk.org; Dumitrescu, Cristian <cristian.dumitrescu@intel.com>
> Cc: Pattan, Reshma <reshma.pattan@intel.com>
> Subject: [PATCH v2] net/softnic: add support for flow API vxlan encap action
> 
> Added support to handle vxlan encap action of rte flow rule.
> 
> Signed-off-by: Reshma Pattan <reshma.pattan@intel.com>
> Acked-by: Dumitrescu Cristian <cristian.dumitrescu@intel.com>
> ---
>  drivers/net/softnic/rte_eth_softnic_flow.c | 192
> +++++++++++++++++++++
>  1 file changed, 192 insertions(+)
> 
> diff --git a/drivers/net/softnic/rte_eth_softnic_flow.c
> b/drivers/net/softnic/rte_eth_softnic_flow.c
> index 23ef32969..3a17046b4 100644
> --- a/drivers/net/softnic/rte_eth_softnic_flow.c
> +++ b/drivers/net/softnic/rte_eth_softnic_flow.c
> @@ -1634,6 +1634,198 @@ flow_rule_action_get(struct pmd_internals
> *softnic,
>  			break;
>  		} /* RTE_FLOW_ACTION_TYPE_METER */
> 
> +		case RTE_FLOW_ACTION_TYPE_VXLAN_ENCAP:
> +		{
> +			const struct rte_flow_action_vxlan_encap *conf =
> +				action->conf;
> +			const struct rte_flow_item *item;
> +			union flow_item spec, mask;
> +			int disabled = 0, status;
> +			size_t size;
> +
> +			if (conf == NULL)
> +				return rte_flow_error_set(error,
> +					EINVAL,
> +					RTE_FLOW_ERROR_TYPE_ACTION,
> +					action,
> +					"VXLAN ENCAP: Null configuration");
> +
> +			item = conf->definition;
> +			if (item == NULL)
> +				return rte_flow_error_set(error,
> +					EINVAL,
> +					RTE_FLOW_ERROR_TYPE_ACTION,
> +					action,
> +					"VXLAN ENCAP: Null configuration
> definition");
> +
> +			if (!(params->action_mask &
> +					(1LLU <<
> RTE_TABLE_ACTION_ENCAP)))
> +				return rte_flow_error_set(error,
> +					EINVAL,
> +
> 	RTE_FLOW_ERROR_TYPE_UNSPECIFIED,
> +					NULL,
> +					"VXLAN ENCAP: Encap action not
> enabled for this table");
> +
> +			/* Check for Ether. */
> +			flow_item_skip_void(&item);
> +			status = flow_item_proto_preprocess(item, &spec,
> &mask,
> +				&size, &disabled, error);
> +			if (status)
> +				return status;
> +
> +			if (item->type != RTE_FLOW_ITEM_TYPE_ETH) {
> +				return rte_flow_error_set(error,
> +					EINVAL,
> +					RTE_FLOW_ERROR_TYPE_ITEM,
> +					item,
> +					"VXLAN ENCAP: first encap item
> should be ether");
> +			}
> +			ether_addr_copy(&spec.eth.dst,
> +					&rule_action-
> >encap.vxlan.ether.da);
> +			ether_addr_copy(&spec.eth.src,
> +					&rule_action->encap.vxlan.ether.sa);
> +
> +			item++;
> +
> +			/* Check for VLAN. */
> +			flow_item_skip_void(&item);
> +			status = flow_item_proto_preprocess(item, &spec,
> &mask,
> +					&size, &disabled, error);
> +			if (status)
> +				return status;
> +
> +			if (item->type == RTE_FLOW_ITEM_TYPE_VLAN) {
> +				if (!params->encap.vxlan.vlan)
> +					return rte_flow_error_set(error,
> +						ENOTSUP,
> +
> 	RTE_FLOW_ERROR_TYPE_ITEM,
> +						item,
> +						"VXLAN ENCAP: vlan encap
> not supported by table");
> +
> +				uint16_t tci = rte_ntohs(spec.vlan.tci);
> +				rule_action->encap.vxlan.vlan.pcp =
> +					tci & 0x7;
> +				rule_action->encap.vxlan.vlan.dei =
> +					(tci & 0x8) >> 3;
> +				rule_action->encap.vxlan.vlan.vid =
> +					tci & 0xfff0;

rule_action->encap.vxlan.vlan.pcp = tci >> 13;
rule_action->encap.vxlan.vlan.dei = (tci >> 12) & 0x1;
rule_action->encap.vxlan.vlan.vid = tci & 0xFFF;

> +
> +				item++;
> +
> +				flow_item_skip_void(&item);
> +				status = flow_item_proto_preprocess(item,
> &spec,
> +						&mask, &size, &disabled,
> error);
> +				if (status)
> +					return status;
> +			} else {
> +				if (params->encap.vxlan.vlan)
> +					return rte_flow_error_set(error,
> +						ENOTSUP,
> +
> 	RTE_FLOW_ERROR_TYPE_ITEM,
> +						item,
> +						"VXLAN ENCAP: expecting
> vlan encap item");
> +			}
> +
> +			/* Check for IPV4/IPV6. */
> +			switch (item->type) {
> +			case RTE_FLOW_ITEM_TYPE_IPV4:
> +			{
> +				rule_action->encap.vxlan.ipv4.sa =
> +					rte_ntohl(spec.ipv4.hdr.src_addr);
> +				rule_action->encap.vxlan.ipv4.da =
> +					rte_ntohl(spec.ipv4.hdr.dst_addr);
> +				rule_action->encap.vxlan.ipv4.dscp =
> +					spec.ipv4.hdr.type_of_service >> 2;
> +				rule_action->encap.vxlan.ipv4.ttl =
> +					spec.ipv4.hdr.time_to_live;
> +				break;
> +			}
> +			case RTE_FLOW_ITEM_TYPE_IPV6:
> +			{
> +				uint32_t vtc_flow;
> +
> +				memcpy(&rule_action->encap.vxlan.ipv6.sa,
> +						&spec.ipv6.hdr.src_addr,
> +
> 	sizeof(spec.ipv6.hdr.src_addr));
> +				memcpy(&rule_action->encap.vxlan.ipv6.da,
> +						&spec.ipv6.hdr.dst_addr,
> +
> 	sizeof(spec.ipv6.hdr.dst_addr));
> +				vtc_flow = rte_ntohl(spec.ipv6.hdr.vtc_flow);
> +				rule_action->encap.vxlan.ipv6.flow_label =
> +						vtc_flow >> 12;
> +				rule_action->encap.vxlan.ipv6.dscp =
> +						(vtc_flow & 0x00000fc0) >> 6;
> +				rule_action->encap.vxlan.ipv6.hop_limit =
> +					spec.ipv6.hdr.hop_limits;
> +				break;
> +			}
> +			default:
> +				return rte_flow_error_set(error,
> +					EINVAL,
> +					RTE_FLOW_ERROR_TYPE_ITEM,
> +					item,
> +					"VXLAN ENCAP: encap item after
> ether should be ipv4/ipv6");
> +			}
> +
> +			item++;
> +
> +			/* Check for UDP. */
> +			flow_item_skip_void(&item);
> +			status = flow_item_proto_preprocess(item, &spec,
> &mask,
> +					&size, &disabled, error);
> +			if (status)
> +				return status;
> +
> +			if (item->type != RTE_FLOW_ITEM_TYPE_UDP) {
> +				return rte_flow_error_set(error,
> +					EINVAL,
> +					RTE_FLOW_ERROR_TYPE_ITEM,
> +					item,
> +					"VXLAN ENCAP: encap item after
> ipv4/ipv6 should be udp");
> +			}
> +			rule_action->encap.vxlan.udp.sp =
> +				rte_ntohs(spec.udp.hdr.src_port);
> +			rule_action->encap.vxlan.udp.dp =
> +				rte_ntohs(spec.udp.hdr.dst_port);
> +
> +			item++;
> +
> +			/* Check for VXLAN. */
> +			flow_item_skip_void(&item);
> +			status = flow_item_proto_preprocess(item, &spec,
> &mask,
> +					&size, &disabled, error);
> +			if (status)
> +				return status;
> +
> +			if (item->type != RTE_FLOW_ITEM_TYPE_VXLAN) {
> +				return rte_flow_error_set(error,
> +					EINVAL,
> +					RTE_FLOW_ERROR_TYPE_ITEM,
> +					item,
> +					"VXLAN ENCAP: encap item after udp
> should be vxlan");
> +			}
> +			rule_action->encap.vxlan.vxlan.vni =
> +				(spec.vxlan.vni[0] << 16U |
> +					spec.vxlan.vni[1] << 8U
> +					| spec.vxlan.vni[2]);
> +
> +			item++;
> +
> +			/* Check for END. */
> +			flow_item_skip_void(&item);
> +
> +			if (item->type != RTE_FLOW_ITEM_TYPE_END)
> +				return rte_flow_error_set(error,
> +					EINVAL,
> +					RTE_FLOW_ERROR_TYPE_ITEM,
> +					item,
> +					"VXLAN ENCAP: expecting END
> item");
> +
> +			rule_action->encap.type =
> RTE_TABLE_ACTION_ENCAP_VXLAN;
> +			rule_action->action_mask |= 1 <<
> RTE_TABLE_ACTION_ENCAP;
> +			break;
> +		} /* RTE_FLOW_ACTION_TYPE_VXLAN_ENCAP */
> +
>  		default:
>  			return -ENOTSUP;
>  		}
> --
> 2.17.1
  
Cristian Dumitrescu Oct. 15, 2018, 4:18 p.m. UTC | #2
> -----Original Message-----
> From: dev [mailto:dev-bounces@dpdk.org] On Behalf Of Dumitrescu,
> Cristian
> Sent: Monday, October 15, 2018 5:05 PM
> To: Pattan, Reshma <reshma.pattan@intel.com>; dev@dpdk.org
> Subject: Re: [dpdk-dev] [PATCH v2] net/softnic: add support for flow API
> vxlan encap action
> 
> 
> 
> > -----Original Message-----
> > From: Pattan, Reshma
> > Sent: Monday, October 15, 2018 3:42 PM
> > To: dev@dpdk.org; Dumitrescu, Cristian <cristian.dumitrescu@intel.com>
> > Cc: Pattan, Reshma <reshma.pattan@intel.com>
> > Subject: [PATCH v2] net/softnic: add support for flow API vxlan encap
> action
> >
> > Added support to handle vxlan encap action of rte flow rule.
> >
> > Signed-off-by: Reshma Pattan <reshma.pattan@intel.com>
> > Acked-by: Dumitrescu Cristian <cristian.dumitrescu@intel.com>
> > ---
> >  drivers/net/softnic/rte_eth_softnic_flow.c | 192
> > +++++++++++++++++++++
> >  1 file changed, 192 insertions(+)
> >
> > diff --git a/drivers/net/softnic/rte_eth_softnic_flow.c
> > b/drivers/net/softnic/rte_eth_softnic_flow.c
> > index 23ef32969..3a17046b4 100644
> > --- a/drivers/net/softnic/rte_eth_softnic_flow.c
> > +++ b/drivers/net/softnic/rte_eth_softnic_flow.c
> > @@ -1634,6 +1634,198 @@ flow_rule_action_get(struct pmd_internals
> > *softnic,
> >  			break;
> >  		} /* RTE_FLOW_ACTION_TYPE_METER */
> >
> > +		case RTE_FLOW_ACTION_TYPE_VXLAN_ENCAP:
> > +		{
> > +			const struct rte_flow_action_vxlan_encap *conf =
> > +				action->conf;
> > +			const struct rte_flow_item *item;
> > +			union flow_item spec, mask;
> > +			int disabled = 0, status;
> > +			size_t size;
> > +
> > +			if (conf == NULL)
> > +				return rte_flow_error_set(error,
> > +					EINVAL,
> > +					RTE_FLOW_ERROR_TYPE_ACTION,
> > +					action,
> > +					"VXLAN ENCAP: Null configuration");
> > +
> > +			item = conf->definition;
> > +			if (item == NULL)
> > +				return rte_flow_error_set(error,
> > +					EINVAL,
> > +					RTE_FLOW_ERROR_TYPE_ACTION,
> > +					action,
> > +					"VXLAN ENCAP: Null configuration
> > definition");
> > +
> > +			if (!(params->action_mask &
> > +					(1LLU <<
> > RTE_TABLE_ACTION_ENCAP)))
> > +				return rte_flow_error_set(error,
> > +					EINVAL,
> > +
> > 	RTE_FLOW_ERROR_TYPE_UNSPECIFIED,
> > +					NULL,
> > +					"VXLAN ENCAP: Encap action not
> > enabled for this table");
> > +
> > +			/* Check for Ether. */
> > +			flow_item_skip_void(&item);
> > +			status = flow_item_proto_preprocess(item, &spec,
> > &mask,
> > +				&size, &disabled, error);
> > +			if (status)
> > +				return status;
> > +
> > +			if (item->type != RTE_FLOW_ITEM_TYPE_ETH) {
> > +				return rte_flow_error_set(error,
> > +					EINVAL,
> > +					RTE_FLOW_ERROR_TYPE_ITEM,
> > +					item,
> > +					"VXLAN ENCAP: first encap item
> > should be ether");
> > +			}
> > +			ether_addr_copy(&spec.eth.dst,
> > +					&rule_action-
> > >encap.vxlan.ether.da);
> > +			ether_addr_copy(&spec.eth.src,
> > +					&rule_action->encap.vxlan.ether.sa);
> > +
> > +			item++;
> > +
> > +			/* Check for VLAN. */
> > +			flow_item_skip_void(&item);
> > +			status = flow_item_proto_preprocess(item, &spec,
> > &mask,
> > +					&size, &disabled, error);
> > +			if (status)
> > +				return status;
> > +
> > +			if (item->type == RTE_FLOW_ITEM_TYPE_VLAN) {
> > +				if (!params->encap.vxlan.vlan)
> > +					return rte_flow_error_set(error,
> > +						ENOTSUP,
> > +
> > 	RTE_FLOW_ERROR_TYPE_ITEM,
> > +						item,
> > +						"VXLAN ENCAP: vlan encap
> > not supported by table");
> > +
> > +				uint16_t tci = rte_ntohs(spec.vlan.tci);
> > +				rule_action->encap.vxlan.vlan.pcp =
> > +					tci & 0x7;
> > +				rule_action->encap.vxlan.vlan.dei =
> > +					(tci & 0x8) >> 3;
> > +				rule_action->encap.vxlan.vlan.vid =
> > +					tci & 0xfff0;
> 
> rule_action->encap.vxlan.vlan.pcp = tci >> 13;
> rule_action->encap.vxlan.vlan.dei = (tci >> 12) & 0x1;
> rule_action->encap.vxlan.vlan.vid = tci & 0xFFF;
> 
> > +
> > +				item++;
> > +
> > +				flow_item_skip_void(&item);
> > +				status = flow_item_proto_preprocess(item,
> > &spec,
> > +						&mask, &size, &disabled,
> > error);
> > +				if (status)
> > +					return status;
> > +			} else {
> > +				if (params->encap.vxlan.vlan)
> > +					return rte_flow_error_set(error,
> > +						ENOTSUP,
> > +
> > 	RTE_FLOW_ERROR_TYPE_ITEM,
> > +						item,
> > +						"VXLAN ENCAP: expecting
> > vlan encap item");
> > +			}
> > +
> > +			/* Check for IPV4/IPV6. */
> > +			switch (item->type) {
> > +			case RTE_FLOW_ITEM_TYPE_IPV4:
> > +			{
> > +				rule_action->encap.vxlan.ipv4.sa =
> > +					rte_ntohl(spec.ipv4.hdr.src_addr);
> > +				rule_action->encap.vxlan.ipv4.da =
> > +					rte_ntohl(spec.ipv4.hdr.dst_addr);
> > +				rule_action->encap.vxlan.ipv4.dscp =
> > +					spec.ipv4.hdr.type_of_service >> 2;
> > +				rule_action->encap.vxlan.ipv4.ttl =
> > +					spec.ipv4.hdr.time_to_live;
> > +				break;
> > +			}
> > +			case RTE_FLOW_ITEM_TYPE_IPV6:
> > +			{
> > +				uint32_t vtc_flow;
> > +
> > +				memcpy(&rule_action->encap.vxlan.ipv6.sa,
> > +						&spec.ipv6.hdr.src_addr,
> > +
> > 	sizeof(spec.ipv6.hdr.src_addr));
> > +				memcpy(&rule_action->encap.vxlan.ipv6.da,
> > +						&spec.ipv6.hdr.dst_addr,
> > +
> > 	sizeof(spec.ipv6.hdr.dst_addr));
> > +				vtc_flow = rte_ntohl(spec.ipv6.hdr.vtc_flow);
> > +				rule_action->encap.vxlan.ipv6.flow_label =
> > +						vtc_flow >> 12;

rule_action->encap.vxlan.ipv6.flow_label = vtc_flow & 0xFFFFF;

> > +				rule_action->encap.vxlan.ipv6.dscp =
> > +						(vtc_flow & 0x00000fc0) >> 6;

rule_action->encap.vxlan.ipv6.dscp = (vtc_flow >> 22) & 0x3F;

> > +				rule_action->encap.vxlan.ipv6.hop_limit =
> > +					spec.ipv6.hdr.hop_limits;
> > +				break;
> > +			}
> > +			default:
> > +				return rte_flow_error_set(error,
> > +					EINVAL,
> > +					RTE_FLOW_ERROR_TYPE_ITEM,
> > +					item,
> > +					"VXLAN ENCAP: encap item after
> > ether should be ipv4/ipv6");
> > +			}
> > +
> > +			item++;
> > +
> > +			/* Check for UDP. */
> > +			flow_item_skip_void(&item);
> > +			status = flow_item_proto_preprocess(item, &spec,
> > &mask,
> > +					&size, &disabled, error);
> > +			if (status)
> > +				return status;
> > +
> > +			if (item->type != RTE_FLOW_ITEM_TYPE_UDP) {
> > +				return rte_flow_error_set(error,
> > +					EINVAL,
> > +					RTE_FLOW_ERROR_TYPE_ITEM,
> > +					item,
> > +					"VXLAN ENCAP: encap item after
> > ipv4/ipv6 should be udp");
> > +			}
> > +			rule_action->encap.vxlan.udp.sp =
> > +				rte_ntohs(spec.udp.hdr.src_port);
> > +			rule_action->encap.vxlan.udp.dp =
> > +				rte_ntohs(spec.udp.hdr.dst_port);
> > +
> > +			item++;
> > +
> > +			/* Check for VXLAN. */
> > +			flow_item_skip_void(&item);
> > +			status = flow_item_proto_preprocess(item, &spec,
> > &mask,
> > +					&size, &disabled, error);
> > +			if (status)
> > +				return status;
> > +
> > +			if (item->type != RTE_FLOW_ITEM_TYPE_VXLAN) {
> > +				return rte_flow_error_set(error,
> > +					EINVAL,
> > +					RTE_FLOW_ERROR_TYPE_ITEM,
> > +					item,
> > +					"VXLAN ENCAP: encap item after udp
> > should be vxlan");
> > +			}
> > +			rule_action->encap.vxlan.vxlan.vni =
> > +				(spec.vxlan.vni[0] << 16U |
> > +					spec.vxlan.vni[1] << 8U
> > +					| spec.vxlan.vni[2]);
> > +
> > +			item++;
> > +
> > +			/* Check for END. */
> > +			flow_item_skip_void(&item);
> > +
> > +			if (item->type != RTE_FLOW_ITEM_TYPE_END)
> > +				return rte_flow_error_set(error,
> > +					EINVAL,
> > +					RTE_FLOW_ERROR_TYPE_ITEM,
> > +					item,
> > +					"VXLAN ENCAP: expecting END
> > item");
> > +
> > +			rule_action->encap.type =
> > RTE_TABLE_ACTION_ENCAP_VXLAN;
> > +			rule_action->action_mask |= 1 <<
> > RTE_TABLE_ACTION_ENCAP;
> > +			break;
> > +		} /* RTE_FLOW_ACTION_TYPE_VXLAN_ENCAP */
> > +
> >  		default:
> >  			return -ENOTSUP;
> >  		}
> > --
> > 2.17.1
  

Patch

diff --git a/drivers/net/softnic/rte_eth_softnic_flow.c b/drivers/net/softnic/rte_eth_softnic_flow.c
index 23ef32969..3a17046b4 100644
--- a/drivers/net/softnic/rte_eth_softnic_flow.c
+++ b/drivers/net/softnic/rte_eth_softnic_flow.c
@@ -1634,6 +1634,198 @@  flow_rule_action_get(struct pmd_internals *softnic,
 			break;
 		} /* RTE_FLOW_ACTION_TYPE_METER */
 
+		case RTE_FLOW_ACTION_TYPE_VXLAN_ENCAP:
+		{
+			const struct rte_flow_action_vxlan_encap *conf =
+				action->conf;
+			const struct rte_flow_item *item;
+			union flow_item spec, mask;
+			int disabled = 0, status;
+			size_t size;
+
+			if (conf == NULL)
+				return rte_flow_error_set(error,
+					EINVAL,
+					RTE_FLOW_ERROR_TYPE_ACTION,
+					action,
+					"VXLAN ENCAP: Null configuration");
+
+			item = conf->definition;
+			if (item == NULL)
+				return rte_flow_error_set(error,
+					EINVAL,
+					RTE_FLOW_ERROR_TYPE_ACTION,
+					action,
+					"VXLAN ENCAP: Null configuration definition");
+
+			if (!(params->action_mask &
+					(1LLU << RTE_TABLE_ACTION_ENCAP)))
+				return rte_flow_error_set(error,
+					EINVAL,
+					RTE_FLOW_ERROR_TYPE_UNSPECIFIED,
+					NULL,
+					"VXLAN ENCAP: Encap action not enabled for this table");
+
+			/* Check for Ether. */
+			flow_item_skip_void(&item);
+			status = flow_item_proto_preprocess(item, &spec, &mask,
+				&size, &disabled, error);
+			if (status)
+				return status;
+
+			if (item->type != RTE_FLOW_ITEM_TYPE_ETH) {
+				return rte_flow_error_set(error,
+					EINVAL,
+					RTE_FLOW_ERROR_TYPE_ITEM,
+					item,
+					"VXLAN ENCAP: first encap item should be ether");
+			}
+			ether_addr_copy(&spec.eth.dst,
+					&rule_action->encap.vxlan.ether.da);
+			ether_addr_copy(&spec.eth.src,
+					&rule_action->encap.vxlan.ether.sa);
+
+			item++;
+
+			/* Check for VLAN. */
+			flow_item_skip_void(&item);
+			status = flow_item_proto_preprocess(item, &spec, &mask,
+					&size, &disabled, error);
+			if (status)
+				return status;
+
+			if (item->type == RTE_FLOW_ITEM_TYPE_VLAN) {
+				if (!params->encap.vxlan.vlan)
+					return rte_flow_error_set(error,
+						ENOTSUP,
+						RTE_FLOW_ERROR_TYPE_ITEM,
+						item,
+						"VXLAN ENCAP: vlan encap not supported by table");
+
+				uint16_t tci = rte_ntohs(spec.vlan.tci);
+				rule_action->encap.vxlan.vlan.pcp =
+					tci & 0x7;
+				rule_action->encap.vxlan.vlan.dei =
+					(tci & 0x8) >> 3;
+				rule_action->encap.vxlan.vlan.vid =
+					tci & 0xfff0;
+
+				item++;
+
+				flow_item_skip_void(&item);
+				status = flow_item_proto_preprocess(item, &spec,
+						&mask, &size, &disabled, error);
+				if (status)
+					return status;
+			} else {
+				if (params->encap.vxlan.vlan)
+					return rte_flow_error_set(error,
+						ENOTSUP,
+						RTE_FLOW_ERROR_TYPE_ITEM,
+						item,
+						"VXLAN ENCAP: expecting vlan encap item");
+			}
+
+			/* Check for IPV4/IPV6. */
+			switch (item->type) {
+			case RTE_FLOW_ITEM_TYPE_IPV4:
+			{
+				rule_action->encap.vxlan.ipv4.sa =
+					rte_ntohl(spec.ipv4.hdr.src_addr);
+				rule_action->encap.vxlan.ipv4.da =
+					rte_ntohl(spec.ipv4.hdr.dst_addr);
+				rule_action->encap.vxlan.ipv4.dscp =
+					spec.ipv4.hdr.type_of_service >> 2;
+				rule_action->encap.vxlan.ipv4.ttl =
+					spec.ipv4.hdr.time_to_live;
+				break;
+			}
+			case RTE_FLOW_ITEM_TYPE_IPV6:
+			{
+				uint32_t vtc_flow;
+
+				memcpy(&rule_action->encap.vxlan.ipv6.sa,
+						&spec.ipv6.hdr.src_addr,
+						sizeof(spec.ipv6.hdr.src_addr));
+				memcpy(&rule_action->encap.vxlan.ipv6.da,
+						&spec.ipv6.hdr.dst_addr,
+						sizeof(spec.ipv6.hdr.dst_addr));
+				vtc_flow = rte_ntohl(spec.ipv6.hdr.vtc_flow);
+				rule_action->encap.vxlan.ipv6.flow_label =
+						vtc_flow >> 12;
+				rule_action->encap.vxlan.ipv6.dscp =
+						(vtc_flow & 0x00000fc0) >> 6;
+				rule_action->encap.vxlan.ipv6.hop_limit =
+					spec.ipv6.hdr.hop_limits;
+				break;
+			}
+			default:
+				return rte_flow_error_set(error,
+					EINVAL,
+					RTE_FLOW_ERROR_TYPE_ITEM,
+					item,
+					"VXLAN ENCAP: encap item after ether should be ipv4/ipv6");
+			}
+
+			item++;
+
+			/* Check for UDP. */
+			flow_item_skip_void(&item);
+			status = flow_item_proto_preprocess(item, &spec, &mask,
+					&size, &disabled, error);
+			if (status)
+				return status;
+
+			if (item->type != RTE_FLOW_ITEM_TYPE_UDP) {
+				return rte_flow_error_set(error,
+					EINVAL,
+					RTE_FLOW_ERROR_TYPE_ITEM,
+					item,
+					"VXLAN ENCAP: encap item after ipv4/ipv6 should be udp");
+			}
+			rule_action->encap.vxlan.udp.sp =
+				rte_ntohs(spec.udp.hdr.src_port);
+			rule_action->encap.vxlan.udp.dp =
+				rte_ntohs(spec.udp.hdr.dst_port);
+
+			item++;
+
+			/* Check for VXLAN. */
+			flow_item_skip_void(&item);
+			status = flow_item_proto_preprocess(item, &spec, &mask,
+					&size, &disabled, error);
+			if (status)
+				return status;
+
+			if (item->type != RTE_FLOW_ITEM_TYPE_VXLAN) {
+				return rte_flow_error_set(error,
+					EINVAL,
+					RTE_FLOW_ERROR_TYPE_ITEM,
+					item,
+					"VXLAN ENCAP: encap item after udp should be vxlan");
+			}
+			rule_action->encap.vxlan.vxlan.vni =
+				(spec.vxlan.vni[0] << 16U |
+					spec.vxlan.vni[1] << 8U
+					| spec.vxlan.vni[2]);
+
+			item++;
+
+			/* Check for END. */
+			flow_item_skip_void(&item);
+
+			if (item->type != RTE_FLOW_ITEM_TYPE_END)
+				return rte_flow_error_set(error,
+					EINVAL,
+					RTE_FLOW_ERROR_TYPE_ITEM,
+					item,
+					"VXLAN ENCAP: expecting END item");
+
+			rule_action->encap.type = RTE_TABLE_ACTION_ENCAP_VXLAN;
+			rule_action->action_mask |= 1 << RTE_TABLE_ACTION_ENCAP;
+			break;
+		} /* RTE_FLOW_ACTION_TYPE_VXLAN_ENCAP */
+
 		default:
 			return -ENOTSUP;
 		}