From mboxrd@z Thu Jan 1 00:00:00 1970 Return-Path: Received: from dpdk.org (dpdk.org [92.243.14.124]) by inbox.dpdk.org (Postfix) with ESMTP id 6295AA04B1; Wed, 4 Nov 2020 14:18:26 +0100 (CET) Received: from [92.243.14.124] (localhost [127.0.0.1]) by dpdk.org (Postfix) with ESMTP id C5474C938; Wed, 4 Nov 2020 14:17:52 +0100 (CET) Received: from mellanox.co.il (mail-il-dmz.mellanox.com [193.47.165.129]) by dpdk.org (Postfix) with ESMTP id 88385C92A for ; Wed, 4 Nov 2020 14:17:50 +0100 (CET) Received: from Internal Mail-Server by MTLPINE1 (envelope-from orika@nvidia.com) with SMTP; 4 Nov 2020 15:17:47 +0200 Received: from MTL-ORIKA.mtl.com ([172.27.14.169]) by labmailer.mlnx (8.13.8/8.13.8) with ESMTP id 0A4DHhw3025641; Wed, 4 Nov 2020 15:17:46 +0200 From: Ori Kam To: andreyv@nvidia.com, mdr@ashroe.eu Cc: alexr@nvidia.com, andrey.vesnovaty@gmail.com, arybchenko@solarflare.com, dev@dpdk.org, elibr@nvidia.com, ferruh.yigit@intel.com, orika@nvidia.com, ozsh@nvidia.com, roniba@nvidia.com, thomas@nvidia.net, viacheslavo@nvidia.com Date: Wed, 4 Nov 2020 15:17:37 +0200 Message-Id: <20201104131737.10006-3-orika@nvidia.com> X-Mailer: git-send-email 2.25.1 In-Reply-To: <20201104131737.10006-1-orika@nvidia.com> References: <20200909203008.25563-1-andreyv@nvidia.com> <20201104131737.10006-1-orika@nvidia.com> MIME-Version: 1.0 Content-Transfer-Encoding: 8bit Subject: [dpdk-dev] [RFC v3 2/2] ethdev: introduce sft lib X-BeenThere: dev@dpdk.org X-Mailman-Version: 2.1.15 Precedence: list List-Id: DPDK patches and discussions List-Unsubscribe: , List-Archive: List-Post: List-Help: List-Subscribe: , Errors-To: dev-bounces@dpdk.org Sender: "dev" Defines RTE SFT (Stateful Flow Table) APIs for Stateful Flow Table library. Currently, DPDK enables only stateless offloading, using the rte_flow. stateless means that each packet is handled without any knowledge of previous or future packets. As we look at the industry, there is much demand to save a context across packets that belong to a connection. Examples for such applications: - Next-generation firewalls - Intrusion detection/prevention systems (IDS/IPS): Suricata, snort - SW/Virtual Switching: OVS The goals of the SFT library: - Accelerate flow recognition & its context retrieval for further lookaside processing. - Enable context-aware flow handling offload. The solution suggested is to create a lib that will enable saving states between different packets that belong to the same connection. The solution will also enable better HW abstraction than the one we get from using the rte_flow. The reason for this is that saving states is not atomic action like we have in rte_flow and also can't be done fully in HW (The first packets must be seen by the application). Saying the above this lib is based on interacting with the rte_flow but it doesn't replace it or encapsulate it. Key design points. - The SFT should offload as much as possible to HW. - The SFT is designed to work alongside the rte_flow. - The SFT has its own ops that the PMD needs to implement. - The SFT works on 5 tuple + zone (a user-defined value) Basic usage flow: 1. Application insert a flow that matches all eth traffic and have sft action along with jump action. (in the future this jump can be avoided and in doing so saving some jumps, but for the most generic and complete solution we think that allow the application full control of the packet process using rte_flow is better.) 2. Application insert a flow in the target group that matches the packet state. Based on this state the application performs the needed actions. This flow can also be merged with other matching criteria. The application will also add a flow in the target group that will upload to the application any packet with a miss state. 3. First eth packet arrives and is routed to the SFT HW component. since this is the first packet the SFT will have a miss and will mark the packet with miss state and forward it to the target group. 4. The application will pull the packet from the queue and will send it to be processed by the sft lib. 5. The SFT will extract HW packet state and if valid the zone or the flow-id, and report it back to the application. 6. Application will see that this is a new connection, so it will issue SFT command to create a new connection with a selected state. The SFT will create a HW flow, that matches the 5 tuple + zone and sets the state of the packet. The state can be any u8 value, it is the responsibility of the application to match on the value. 7. On the next packet arriving to the HW it will jump to the SFT and in the SFT HW there will be a match which will result in setting the packet state and ID according to the application request. 8. In case of later miss (at some other group) or application logic that the packet should be routed back to the application. The application will call the SFT lib with the new mbuf, which will result in the flow-id returned to the application along with the context attached to this connection. Signed-off-by: Andrey Vesnovaty Signed-off-by: Ori Kam --- lib/librte_ethdev/meson.build | 3 + lib/librte_ethdev/rte_ethdev_version.map | 19 + lib/librte_ethdev/rte_sft.c | 9 + lib/librte_ethdev/rte_sft.h | 877 +++++++++++++++++++++++ lib/librte_ethdev/rte_sft_driver.h | 201 ++++++ 5 files changed, 1109 insertions(+) create mode 100644 lib/librte_ethdev/rte_sft.c create mode 100644 lib/librte_ethdev/rte_sft.h create mode 100644 lib/librte_ethdev/rte_sft_driver.h diff --git a/lib/librte_ethdev/meson.build b/lib/librte_ethdev/meson.build index 8fc24e8c8a..064e3c9443 100644 --- a/lib/librte_ethdev/meson.build +++ b/lib/librte_ethdev/meson.build @@ -9,6 +9,7 @@ sources = files('ethdev_private.c', 'rte_ethdev.c', 'rte_flow.c', 'rte_mtr.c', + 'rte_sft.c', 'rte_tm.c') headers = files('rte_ethdev.h', @@ -24,6 +25,8 @@ headers = files('rte_ethdev.h', 'rte_flow_driver.h', 'rte_mtr.h', 'rte_mtr_driver.h', + 'rte_sft.h', + 'rte_sft_driver.h', 'rte_tm.h', 'rte_tm_driver.h') diff --git a/lib/librte_ethdev/rte_ethdev_version.map b/lib/librte_ethdev/rte_ethdev_version.map index f8a0945812..e3c829b494 100644 --- a/lib/librte_ethdev/rte_ethdev_version.map +++ b/lib/librte_ethdev/rte_ethdev_version.map @@ -232,6 +232,25 @@ EXPERIMENTAL { rte_eth_fec_get_capability; rte_eth_fec_get; rte_eth_fec_set; + rte_sft_drain_mbuf; + rte_sft_fini; + rte_sft_flow_activate; + rte_sft_flow_create; + rte_sft_flow_destroy; + rte_sft_flow_get_client_obj; + rte_sft_flow_get_status; + rte_sft_flow_query; + rte_sft_flow_set_aging; + rte_sft_flow_set_client_obj; + rte_sft_flow_set_data; + rte_sft_flow_set_offload; + rte_sft_flow_set_state; + rte_sft_flow_touch; + rte_sft_init; + rte_sft_process_mbuf; + rte_sft_process_mbuf_with_zone; + + }; INTERNAL { diff --git a/lib/librte_ethdev/rte_sft.c b/lib/librte_ethdev/rte_sft.c new file mode 100644 index 0000000000..f3d3945545 --- /dev/null +++ b/lib/librte_ethdev/rte_sft.c @@ -0,0 +1,9 @@ +/* SPDX-License-Identifier: BSD-3-Clause + * Copyright 2020 Mellanox Technologies, Ltd + */ + + +#include "rte_sft.h" +#include "rte_sft_driver.h" + +/* Placeholder for RTE SFT library APIs implementation */ diff --git a/lib/librte_ethdev/rte_sft.h b/lib/librte_ethdev/rte_sft.h new file mode 100644 index 0000000000..d295bb0b7a --- /dev/null +++ b/lib/librte_ethdev/rte_sft.h @@ -0,0 +1,877 @@ +/* SPDX-License-Identifier: BSD-3-Clause + * Copyright 2020 Mellanox Technologies, Ltd + */ + +#ifndef _RTE_SFT_H_ +#define _RTE_SFT_H_ + +/** + * @file + * + * RTE SFT API + * + * Defines RTE SFT APIs for Statefull Flow Table library. + * + * The SFT lib is part of the ethdev class, the reason for this is that the main + * idea is to leverage the HW offload that the ethdev allow using the rte_flow. + * + * SFT General description: + * SFT library provides a framework for applications that need to maintain + * context across different packets of the connection. + * Examples for such applications: + * - Next-generation firewalls + * - Intrusion detection/prevention systems (IDS/IPS): Suricata, Snort + * - SW/Virtual Switching: OVS + * The goals of the SFT library: + * - Accelerate flow recognition & its context retrieval for further look-aside + * processing. + * - Enable context-aware flow handling offload. + * + * The SFT is designed to use HW offload to get the best performance. + * This is done on two levels. The first one is marking the packet with flow id + * to speed the lookup of the flow in the data structure. + * The second is done be connecting the SFT results to the rte_flow for + * continuing packet process. + * + * Definitions and Abbreviations: + * - 5-tuple: defined by: + * -- Source IP address + * -- Source port + * -- Destination IP address + * -- Destination port + * -- IP protocol number + * - 7-tuple: 5-tuple, zone and port (see struct rte_sft_7tuple) + * - 5/7-tuple: 5/7-tuple of the packet from connection initiator + * - revers 5/7-tuple: 5/7-tuple of the packet from connection initiate + * - application: SFT library API consumer + * - APP: see application + * - CID: client ID + * - CT: connection tracking + * - FID: Flow identifier + * - FIF: First In Flow + * - Flow: defined by 7-tuple and its reverse i.e. flow is bidirectional + * - SFT: Stateful Flow Table + * - user: see application + * - zone: additional user defined value used as differentiator for + * connections having same 5-tuple (for example different VXLAN + * connections with same inner 5-tuple). + * + * SFT components: + * + * +-----------------------------------+ + * | RTE flow | + * | | + * | +-------------------------------+ | +----------------+ + * | | group X | | | RTE_SFT | + * | | | | | | + * | | +---------------------------+ | | | | + * | | | rule ... | | | | | + * | | | . | | | +-----------+----+ + * | | | . | | | | + * | | | . | | | entry + * | | +---------------------------+ | | create + * | | | rule | | | | + * | | | patterns ... +---------+ | + * | | | actions | | | | | + * | | | SFT (zone=Z) | | | | | + * | | | JUMP (group=Y) | | | lookup | + * | | +---------------------------+ | | zone=Z, | + * | | | rule ... | | | 5tuple | + * | | | . | | | | | + * | | | . | | | +--v-------------+ + * | | | . | | | | SFT | | + * | | | | | | | | | + * | | +---------------------------+ | | | +--v--+ | + * | | | | | | | | + * | +-------------------------------+ | | | PMD | | + * | | | | | | + * | | | +-----+ | + * | +-------------------------------+ | | | + * | | group Y | | | | + * | | | | | set state | + * | | +---------------------------+ | | | set data | + * | | | rule | | | +--------+-------+ + * | | | patterns | | | | + * | | | SFT (state=UNDEFINED) | | | | + * | | | actions RSS | | | | + * | | +---------------------------+ | | | + * | | | rule | | | | + * | | | patterns | | | | + * | | | SFT (state=INVALID) | <-------------+ + * | | | actions DROP | | | forward + * | | +---------------------------+ | | group=Y + * | | | rule | | | + * | | | patterns | | | + * | | | SFT (state=ACCEPTED) | | | + * | | | actions PORT | | | + * | | +---------------------------+ | | + * | | ... | | + * | | | | + * | +-------------------------------+ | + * | ... | + * | | + * +-----------------------------------+ + * + * SFT as datastructure: + * SFT can be treated as datastructure maintaining flow context across its + * lifetime. SFT flow entry represents bidirectional network flow and defined by + * 7-tuple & its reverse 7-tuple. + * Each entry in SFT has: + * - FID: 1:1 mapped & used as entry handle & encapsulating internal + * implementation of the entry. + * - State: user-defined value attached to each entry, the only library + * reserved value for state unset (the actual value defined by SFT + * configuration). The application should define flow state encodings and + * set it for flow via rte_sft_flow_set_ctx() than what actions should be + * applied on packets can be defined via related RTE flow rule matching SFT + * state (see rules in SFT components diagram above). + * - Timestamp: for the last seen in flow packet used for flow aging mechanism + * implementation. + * - Client Objects: user-defined flow contexts attached as opaques to flow. + * - Acceleration & offloading - utilize RTE flow capabilities, when supported + * (see action ``SFT``), for flow lookup acceleration and further + * context-aware flow handling offload. + * - CT state: optionally for TCP connections CT state can be maintained + * (see enum rte_sft_flow_ct_state). + * - Out of order TCP packets: optionally SFT can keep out of order TCP + * packets aside the flow context till the arrival of the missing in-order + * packet. + * + * RTE flow changes: + * The SFT flow state (or context) for RTE flow is defined by fields of + * struct rte_flow_item_sft. + * To utilize SFT capabilities new item and action types introduced: + * - item SFT: matching on SFT flow state (see RTE_FLOW_ITEM_TYPE_SFT). + * - action SFT: retrieve SFT flow context and attached it to the processed + * packet (see RTE_FLOW_ACTION_TYPE_SFT). + * + * The contents of per port SFT serving RTE flow action ``SFT`` managed via + * SFT PMD APIs (see struct rte_sft_ops). + * The SFT flow state/context retrieval performed by user-defined zone ``SFT`` + * action argument and processed packet 5-tuple. + * If in scope of action ``SFT`` there is no context/state for the flow in SFT + * undefined state attached to the packet meaning that the flow is not + * recognized by SFT, most probably FIF packet. + * + * Once the SFT state set for a packet it can match on item SFT + * (see RTE_FLOW_ITEM_TYPE_SFT) and forwarding design can be done for the + * packet, for example: + * - if state value == x than queue for further processing by the application + * - if state value == y than forward it to eth port (full offload) + * - if state value == 'undefined' than queue for further processing by + * the application (handle FIF packets) + * + * Processing packets with SFT library: + * + * FIF packet: + * To recognize upcoming packets of the SFT flow every FIF packet should be + * forwarded to the application utilizing the SFT library. Non-FIF packets can + * be processed by the application or its processing can be fully offloaded. + * Processing of the packets in SFT library starts with rte_sft_process_mbuf + * or rte_sft_process_mbuf_with_zone. If mbuf recognized as FIF application + * should make a design to destroy flow or complete flow creation process in + * SFT using rte_sft_flow_activate. + * + * Recognized SFT flow: + * Once struct rte_sft_flow_status with valid fid field possessed by application + * it can: + * - mange client objects on it (see client_obj field in + * struct rte_sft_flow_status) using rte_sft_flow__client_obj APIs + * - analyze user-defined flow state and CT state. + * - set flow state to be attached to the upcoming packets by action ``SFT`` + * via struct rte_sft_flow_status API. + * - decide to destroy flow via rte_sft_flow_destroy API. + * + * Flow aging: + * + * SFT library manages the aging for each flow. On flow creation, it's + * assigned an aging value, the maximal number of seconds passed since the + * last flow packet arrived, once exceeded flow considered aged. + * The application notified of aged flow asynchronously via event queues. + * The device and port IDs tuple to identify the event queue to enqueue + * flow aged events passed on flow creation as arguments + * (see rte_sft_flow_activate). It's the application responsibility to + * initialize event queues and assign them to each flow for EOF event + * notifications. + * Aged EOF event handling: + * - Should be considered as application responsibility. + * - The last stage should be the release of the flow resources via + * rte_sft_flow_destroy API. + * - All client objects should be removed from flow before the + * rte_sft_flow_destroy API call. + * See the description of ret_sft_flow_destroy for an example of aged flow + * handling. + * + * SFT API thread safety: + * + * Since the SFT lib is designed to work as part of the Fast-Path, The SFT + * is not thread safe, in order to enable better working with multiple threads + * the SFT lib uses the queue approach, where each queue can only be accessesd + * by one thread while one thread can access multiple queues. + * + * SFT Library initialization and cleanup: + * + * SFT library should be considered as a single instance, preconfigured and + * initialized via rte_sft_init() API. + * SFT library resource deallocation and cleanup should be done via + * rte_sft_init() API as a stage of the application termination procedure. + */ + +#ifdef __cplusplus +extern "C" { +#endif + +#include +#include +#include +#include +#include +#include + +/** + * L3/L4 5-tuple - src/dest IP and port and IP protocol. + * + * Used for flow/connection identification. + */ +RTE_STD_C11 +struct rte_sft_5tuple { + union { + struct { + rte_be32_t src_addr; /**< IPv4 source address. */ + rte_be32_t dst_addr; /**< IPv4 destination address. */ + } ipv4; + struct { + uint8_t src_addr[16]; /**< IPv6 source address. */ + uint8_t dst_addr[16]; /**< IPv6 destination address. */ + } ipv6; + }; + rte_be16_t src_port; /**< Source port. */ + rte_be16_t dst_port; /**< Destination port. */ + uint8_t proto; /**< IP protocol. */ + uint8_t is_ipv6: 1; /**< True for valid IPv6 fields. Otherwise IPv4. */ +}; + +/** + * Port flow identification. + * + * @p zone used for setups where 5-tuple is not enough to identify flow. + * For example different VLANs/VXLANs may have similar 5-tuples. + */ +struct rte_sft_7tuple { + struct rte_sft_5tuple flow_5tuple; /**< L3/L4 5-tuple. */ + uint32_t zone; /**< Zone assigned to flow. */ + uint16_t port_id; /** nb_in_order_mbufs output param !=0 and until + * status->nb_in_order_mbufs == 0. + * Flow should be locked by caller (see rte_sft_flow_lock). + * + * @param queue + * The sft queue number. + * @param fid + * SFT flow ID. + * @param[out] mbuf_out + * last processed not fragmented and in order mbuf. + * @param nb_out + * Number of buffers to be drained. + * @param initiator + * true packets that will be drained belongs to the initiator. + * @param[out] status + * Connection status based on the last mbuf that was drained. + * @param[out] error + * Perform verbose error reporting if not NULL. Initialize in case of + * error only. + * + * @return + * The number of mbufs that were drained, negative value in case + * of error and rte_sft_error is set. + */ +__rte_experimental +int +rte_sft_drain_mbuf(uint16_t queue, uint32_t fid, struct rte_mbuf **mbuf_out, + uint16_t nb_out, bool initiator, + struct rte_sft_flow_status *status, + struct rte_sft_error *error); + +/** + * @warning + * @b EXPERIMENTAL: this API may change without prior notice. + * + * Activate flow in SFT. + * + * This function creates an entry in the SFT for this connection. + * The reasons for 2 phase flow creation procedure: + * 1. Missing reverse flow - flow context is shared for both flow directions + * i.e. in order maintain bidirectional flow context in RTE SFT packets + * arriving from both directions should be identified as packets of the + * RTE SFT flow. Consequently, before the creation of the SFT flow caller + * should provide reverse flow direction 7-tuple. + * 2. The caller of rte_sft_process_mbuf/rte_sft_process_mbuf_with_zone should + * be notified that arrived mbuf is first in flow & decide whether to + * create a new flow or disregard this packet. + * This function completes the creation of the bidirectional SFT flow & creates + * entry for 7-tuple on SFT PMD defined by the tuple port for both + * initiator/initiate 7-tuples. + * Flow aging, connection tracking state & out of order handling will be + * initialized according to the content of the *mbuf_in* passes to + * rte_sft_process_mbuf/_with_zone during phase 1 of flow creation. + * Once this function returns upcoming calls rte_sft_process_mbuf/_with_zone + * with 7-tuple or its reverse will return the handle to this flow. + * Flow should be locked by the caller (see rte_sft_flow_lock). + * + * @param queue + * The SFT queue. + * @param[in] mbuf_in + * mbuf to process; mbuf pointer considered 'consumed' and should not be used + * after successful call to this function. + * @param reverse_tuple + * Expected response flow 7-tuple. + * @param state + * User defined state to set. + * @param data + * User defined data, the len is configured during sft init. + * @param proto_enable + * Enables maintenance of status->proto_state connection tracking value + * for the flow. otherwise status->proto_state will be initialized with zeros. + * @param dev_id + * Event dev ID to enqueue end of flow event. + * @param port_id + * Event port ID to enqueue end of flow event. + * @param actions + * Flags that indicate which actions should be done on the packet before + * returning it to the rte_flow. + * @param action_specs + * Hold the actions configuration. + * @param[out] mbuf_out + * last processed not fragmented and in order mbuf. + * @param[out] status + * Structure to dump SFT flow status once activated. + * @param[out] error + * Perform verbose error reporting if not NULL. SFT initialize this + * structure in case of error only. + * + * @return + * 0 on success, a negative errno value otherwise and rte_sft_error is set. + */ +__rte_experimental +int +rte_sft_flow_activate(uint16_t queue, struct rte_mbuf *mbuf_in, + const struct rte_sft_7tuple *reverse_tuple, + uint8_t state, uint32_t *data, uint8_t proto_enable, + uint8_t dev_id, uint8_t port_id, uint64_t actions, + const struct rte_sft_actions_specs *action_specs, + struct rte_mbuf **mbuf_out, + struct rte_sft_flow_status *status, + struct rte_sft_error *error); + +/** + * @warning + * @b EXPERIMENTAL: this API may change without prior notice. + * + * Artificially create SFT flow. + * + * Function to create SFT flow before reception of the first flow packet. + * + * @param queue + * The SFT queue. + * @param tuple + * Expected initiator flow 7-tuple. + * @param reverse_tuple + * Expected initiate flow 7-tuple. + * @param state + * User defined state to set. + * @param data + * User defined data, the len is configured during sft init. + * @param proto_enable + * Enables maintenance of status->proto_state connection tracking value + * for the flow. otherwise status->proto_state will be initialized with zeros. + * @param[out] status + * Connection status. + * @param[out] error + * Perform verbose error reporting if not NULL. PMDs initialize this + * structure in case of error only. + * + * @return + * - on success: 0, locked SFT flow recognized by status->fid. + * - on error: a negative errno value otherwise and rte_errno is set. + */ +__rte_experimental +int +rte_sft_flow_create(uint16_t queue, const struct rte_sft_7tuple *tuple, + const struct rte_sft_7tuple *reverse_tuple, + const struct rte_flow_item_sft *ctx, + uint8_t ct_enable, + struct rte_sft_flow_status *status, + struct rte_sft_error *error); + +/** + * @warning + * @b EXPERIMENTAL: this API may change without prior notice. + * + * Removes flow from SFT. + * + * @param queue + * The SFT queue. + * @param fid + * SFT flow ID to destroy. + * @param[out] error + * Perform verbose error reporting if not NULL. SFT initialize this + * structure in case of error only. + * + * @return + * 0 on success, a negative errno value otherwise and rte_sft_error is set. + */ +__rte_experimental +int +rte_sft_flow_destroy(uint16_t queue, uint32_t fid, struct rte_sft_error *error); + +/** + * @warning + * @b EXPERIMENTAL: this API may change without prior notice. + * + * Query counter and aging data. + * + * @param queue + * The SFT queue. + * @param fid + * SFT flow ID. + * @param[out] data. + * SFT flow ID. + * @param[out] error + * Perform verbose error reporting if not NULL. SFT initialize this + * structure in case of error only. + * + * @return + * 0 on success, a negative errno value otherwise and rte_sft_error is set. + */ +__rte_experimental +int +rte_sft_flow_query(uint16_t queue, uint32_t fid, + struct rte_sft_query_data *data, + struct rte_sft_error *error); + +/** + * @warning + * @b EXPERIMENTAL: this API may change without prior notice. + * + * Reset flow age to zero. + * + * Simulates last flow packet with timestamp set to just now. + * + * @param queue + * The SFT queue. + * @param fid + * SFT flow ID. + * @param[out] error + * Perform verbose error reporting if not NULL. SFT initialize this + * structure in case of error only. + * + * @return + * 0 on success, a negative errno value otherwise and rte_sft_error is set. + */ +__rte_experimental +int +rte_sft_flow_touch(uint16_t queue, uint32_t fid, struct rte_sft_error *error); + +/** + * @warning + * @b EXPERIMENTAL: this API may change without prior notice. + * + * Set flow aging to specific value. + * + * @param queue + * The SFT queue. + * @param fid + * SFT flow ID. + * @param aging + * New flow aging value. + * @param[out] error + * Perform verbose error reporting if not NULL. SFT initialize this + * structure in case of error only. + * + * @return + * 0 on success, a negative errno value otherwise and rte_sft_error is set. + */ +__rte_experimental +int +rte_sft_flow_set_aging(uint16_t queue, uint32_t fid, uint32_t aging, + struct rte_sft_error *error); + +/** + * @warning + * @b EXPERIMENTAL: this API may change without prior notice. + * + * Set client object for given client ID. + * + * @param queue + * The SFT queue. + * @param fid + * SFT flow ID. + * @param client_id + * Client ID to set object for. + * @param client_obj + * Pointer to opaque client object structure. + * @param[out] error + * Perform verbose error reporting if not NULL. SFT initialize this + * structure in case of error only. + * + * @return + * 0 on success, a negative errno value otherwise and rte_sft_error is set. + */ +__rte_experimental +int +rte_sft_flow_set_client_obj(uint16_t queue, uint32_t fid, uint8_t client_id, + void *client_obj, struct rte_sft_error *error); + +/** + * @warning + * @b EXPERIMENTAL: this API may change without prior notice. + * + * Get client object for given client ID. + * + * @param queue + * The SFT queue. + * @param fid + * SFT flow ID. + * @param client_id + * Client ID to get object for. + * @param[out] error + * Perform verbose error reporting if not NULL. SFT initialize this + * structure in case of error only. + * + * @return + * A valid client object opaque pointer in case of success, NULL otherwise + * and rte_sft_error is set. + */ +__rte_experimental +void * +rte_sft_flow_get_client_obj(uint16_t queue, const uint32_t fid, + uint8_t client_id, struct rte_sft_error *error); + + +#ifdef __cplusplus +} +#endif + +#endif /* _RTE_SFT_H_ */ diff --git a/lib/librte_ethdev/rte_sft_driver.h b/lib/librte_ethdev/rte_sft_driver.h new file mode 100644 index 0000000000..6ae3c4b997 --- /dev/null +++ b/lib/librte_ethdev/rte_sft_driver.h @@ -0,0 +1,201 @@ +/* SPDX-License-Identifier: BSD-3-Clause + * Copyright 2020 Mellanox Technologies, Ltd + */ + +#ifndef RTE_SFT_DRIVER_H_ +#define RTE_SFT_DRIVER_H_ + +/** + * @file + * RTE generic SFT API (driver side) + * + * This file provides implementation helpers for internal use by PMDs, they + * are not intended to be exposed to applications and are not subject to ABI + * versioning. + */ + +#include + +#include "rte_ethdev.h" +#include "rte_ethdev_driver.h" +#include "rte_sft.h" +#include "rte_flow.h" + +#ifdef __cplusplus +extern "C" { +#endif + +struct rte_sft_entry; + +#define RTE_SFT_STATE_FLAG_FID_VALID (1 << 0) +#define RTE_SFT_STATE_FLAG_ZONE_VALID (1 << 1) +#define RTE_SFT_STATE_FLAG_FLOW_MISS (1 << 2) + +#define RTE_SFT_MISS_TCP_FLAGS (1 << 0) + +RTE_STD_C11 +struct rte_sft_decode_info { + union { + uint32_t fid; /**< The fid value. */ + uint32_t zone; /**< The zone value. */ + }; + uint32_t state; + /**< Flags that mark the packet state. see RTE_SFT_STATE_FLAG_*. */ +}; + +/** + * @internal + * Insert a flow to the SFT HW component. + * + * @param dev + * ethdev handle of port. + * @param fid + * Flow ID. + * @param queue + * The sft working queue. + * @param pattern + * The matching pattern. + * @param miss_conditions + * The conditions that forces a miss even if the 5 tuple was matched + * see RTE_SFT_MISS_*. + * @param actions + * Set pf actions to apply in case the flow was hit. If no terminating action + * (queue, rss, drop, port) was given, the terminating action should be taken + * from the flow that resulted in the SFT. + * @param miss_actions + * Set pf actions to apply in case the flow was hit. but the miss conditions + * were hit. (6 tuple match but tcp flags are on) If no terminating action + * (queue, rss, drop, port) was given, the terminating action should be taken + * from the flow that resulted in the SFT. + * @param data + * The application data to attached to the flow. + * @param data_len + * The length of the data in uint32_t increments. + * @param state + * The application state to set. + * @param error[out] + * Verbose of the error. + * + * @return + * Pointer to sft_entry in case of success, null otherwise and rte_sft_error + * is set. + */ +typedef struct rte_sft_entry *(*sft_entry_create_t) + (struct rte_eth_dev *dev, uint32_t fid, uint16_t queue, + const struct rte_flow_item *pattern, uint64_t miss_conditions, + const struct rte_flow_action *actions, + const struct rte_flow_action *miss_actions, + const uint32_t *data, uint16_t data_len, uint8_t state, + struct rte_sft_error *error); + +/** + * @internal + * Modify the state and the data of SFT flow in HW component. + * + * @param dev + * ethdev handle of port. + * @param entry + * The entry to modify. + * @param queue + * The sft working queue. + * @param data + * The application data to attached to the flow. + * @param data_len + * The length of the data in uint32_t increments. + * @param state + * The application state to set. + * @param error[out] + * Verbose of the error. + * + * @return + * Negative errno value on error, 0 on success. + */ +typedef int *(*sft_entry_modify_t)(struct rte_eth_dev *dev, + struct rte_sft_entry *entry, uint16_t queue, + const uint32_t *data, uint16_t data_len, + uint8_t state, struct rte_sft_error *error); + +/** + * @internal + * Destroy SFT flow in HW component. + * + * @param dev + * ethdev handle of port. + * @param entry + * The entry to modify. + * @param queue + * The sft working queue. + * @param error[out] + * Verbose of the error. + * + * @return + * Negative errno value on error, 0 on success. + */ +typedef int *(*sft_entry_destroy_t)(struct rte_eth_dev *dev, + struct rte_sft_entry *entry, uint16_t queue, + struct rte_sft_error *error); + +/** + * @internal + * Decode sft state and FID from mbuf. + * + * @param dev + * ethdev handle of port. + * @param entry + * The entry to modify. + * @param queue + * The sft working queue. + * @param mbuf + * The input mbuf. + * @param info[out] + * The decoded sft data. + * @param error[out] + * Verbose of the error. + * + * @return + * Negative errno value on error, 0 on success. + */ +typedef int *(*sft_entry_decode_t)(struct rte_eth_dev *dev, + struct rte_sft_entry *entry, uint16_t queue, + struct rte_mbuf *mbuf, + struct rte_sft_decode_info *info, + struct rte_sft_error *error); + +/** + * Generic sft operations structure implemented and returned by PMDs. + * + * If successful, this operation must result in a pointer to a PMD-specific. + * + * See also rte_sft_ops_get(). + * + * These callback functions are not supposed to be used by applications + * directly, which must rely on the API defined in rte_sft.h. + */ +struct rte_sft_ops { + sft_entry_create_t sft_create_entry; + sft_entry_modify_t sft_entry_modify; + sft_entry_destroy_t sft_entry_destroy; + sft_entry_decode_t sft_entry_decode; +}; + +/** + * Get generic sft operations structure from a port. + * + * @param port_id + * Port identifier to query. + * @param[out] error + * Pointer to flow error structure. + * + * @return + * The flow operations structure associated with port_id, NULL in case of + * error, in which case rte_errno is set and the error structure contains + * additional details. + */ +const struct rte_sft_ops * +rte_sft_ops_get(uint16_t port_id, struct rte_sft_error *error); + +#ifdef __cplusplus +} +#endif + +#endif /* RTE_SFT_DRIVER_H_ */ -- 2.25.1