Commit ee5d032f authored by Jakub Kicinski's avatar Jakub Kicinski Committed by David S. Miller

xdp: add HW offload mode flag for installing programs

Add an installation-time flag for requesting that the program
be installed only if it can be offloaded to HW.

Internally new command for ndo_xdp is added, this way we avoid
putting checks into drivers since they all return -EINVAL on
an unknown command.
Signed-off-by: default avatarJakub Kicinski <jakub.kicinski@netronome.com>
Acked-by: default avatarDaniel Borkmann <daniel@iogearbox.net>
Signed-off-by: default avatarDavid S. Miller <davem@davemloft.net>
parent 32d60277
...@@ -807,6 +807,7 @@ enum xdp_netdev_command { ...@@ -807,6 +807,7 @@ enum xdp_netdev_command {
* when it is no longer used. * when it is no longer used.
*/ */
XDP_SETUP_PROG, XDP_SETUP_PROG,
XDP_SETUP_PROG_HW,
/* Check if a bpf program is set on the device. The callee should /* Check if a bpf program is set on the device. The callee should
* return true if a program is currently attached and running. * return true if a program is currently attached and running.
*/ */
......
...@@ -891,9 +891,12 @@ enum { ...@@ -891,9 +891,12 @@ enum {
#define XDP_FLAGS_UPDATE_IF_NOEXIST (1U << 0) #define XDP_FLAGS_UPDATE_IF_NOEXIST (1U << 0)
#define XDP_FLAGS_SKB_MODE (1U << 1) #define XDP_FLAGS_SKB_MODE (1U << 1)
#define XDP_FLAGS_DRV_MODE (1U << 2) #define XDP_FLAGS_DRV_MODE (1U << 2)
#define XDP_FLAGS_HW_MODE (1U << 3)
#define XDP_FLAGS_MODES (XDP_FLAGS_SKB_MODE | \
XDP_FLAGS_DRV_MODE | \
XDP_FLAGS_HW_MODE)
#define XDP_FLAGS_MASK (XDP_FLAGS_UPDATE_IF_NOEXIST | \ #define XDP_FLAGS_MASK (XDP_FLAGS_UPDATE_IF_NOEXIST | \
XDP_FLAGS_SKB_MODE | \ XDP_FLAGS_MODES)
XDP_FLAGS_DRV_MODE)
/* These are stored into IFLA_XDP_ATTACHED on dump. */ /* These are stored into IFLA_XDP_ATTACHED on dump. */
enum { enum {
......
...@@ -6957,6 +6957,9 @@ static int dev_xdp_install(struct net_device *dev, xdp_op_t xdp_op, ...@@ -6957,6 +6957,9 @@ static int dev_xdp_install(struct net_device *dev, xdp_op_t xdp_op,
struct netdev_xdp xdp; struct netdev_xdp xdp;
memset(&xdp, 0, sizeof(xdp)); memset(&xdp, 0, sizeof(xdp));
if (flags & XDP_FLAGS_HW_MODE)
xdp.command = XDP_SETUP_PROG_HW;
else
xdp.command = XDP_SETUP_PROG; xdp.command = XDP_SETUP_PROG;
xdp.extack = extack; xdp.extack = extack;
xdp.flags = flags; xdp.flags = flags;
...@@ -6985,7 +6988,7 @@ int dev_change_xdp_fd(struct net_device *dev, struct netlink_ext_ack *extack, ...@@ -6985,7 +6988,7 @@ int dev_change_xdp_fd(struct net_device *dev, struct netlink_ext_ack *extack,
ASSERT_RTNL(); ASSERT_RTNL();
xdp_op = xdp_chk = ops->ndo_xdp; xdp_op = xdp_chk = ops->ndo_xdp;
if (!xdp_op && (flags & XDP_FLAGS_DRV_MODE)) if (!xdp_op && (flags & (XDP_FLAGS_DRV_MODE | XDP_FLAGS_HW_MODE)))
return -EOPNOTSUPP; return -EOPNOTSUPP;
if (!xdp_op || (flags & XDP_FLAGS_SKB_MODE)) if (!xdp_op || (flags & XDP_FLAGS_SKB_MODE))
xdp_op = generic_xdp_install; xdp_op = generic_xdp_install;
......
...@@ -16,6 +16,7 @@ ...@@ -16,6 +16,7 @@
* Vitaly E. Lavrov RTA_OK arithmetics was wrong. * Vitaly E. Lavrov RTA_OK arithmetics was wrong.
*/ */
#include <linux/bitops.h>
#include <linux/errno.h> #include <linux/errno.h>
#include <linux/module.h> #include <linux/module.h>
#include <linux/types.h> #include <linux/types.h>
...@@ -2253,8 +2254,7 @@ static int do_setlink(const struct sk_buff *skb, ...@@ -2253,8 +2254,7 @@ static int do_setlink(const struct sk_buff *skb,
err = -EINVAL; err = -EINVAL;
goto errout; goto errout;
} }
if ((xdp_flags & XDP_FLAGS_SKB_MODE) && if (hweight32(xdp_flags & XDP_FLAGS_MODES) > 1) {
(xdp_flags & XDP_FLAGS_DRV_MODE)) {
err = -EINVAL; err = -EINVAL;
goto errout; goto errout;
} }
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment