Message ID | 20230104194132.24637-7-gerhard@engleder-embedded.com (mailing list archive) |
---|---|
State | Superseded |
Delegated to: | Netdev Maintainers |
Headers | show |
Series | tsnep: XDP support | expand |
From: Gerhard Engleder <gerhard@engleder-embedded.com> Date: Wed Jan 04 2023 20:41:29 GMT+0100 > Implement setup of BPF programs for XDP RX path with command > XDP_SETUP_PROG of ndo_bpf(). This is prework for XDP RX path support. > > tsnep_netdev_close() is called directly during BPF program setup. Add > netif_carrier_off() and netif_tx_stop_all_queues() calls to signal to > network stack that device is down. Otherwise network stack would > continue transmitting pakets. > > Return value of tsnep_netdev_open() is not checked during BPF program > setup like in other drivers. Forwarding the return value would result in > a bpf_prog_put() call in dev_xdp_install(), which would make removal of > BPF program necessary. > > If tsnep_netdev_open() fails during BPF program setup, then the network > stack would call tsnep_netdev_close() anyway. Thus, tsnep_netdev_close() > checks now if device is already down. > > Signed-off-by: Gerhard Engleder <gerhard@engleder-embedded.com> > --- > drivers/net/ethernet/engleder/Makefile | 2 +- > drivers/net/ethernet/engleder/tsnep.h | 13 +++++++++++ > drivers/net/ethernet/engleder/tsnep_main.c | 25 +++++++++++++++++--- > drivers/net/ethernet/engleder/tsnep_xdp.c | 27 ++++++++++++++++++++++ > 4 files changed, 63 insertions(+), 4 deletions(-) > create mode 100644 drivers/net/ethernet/engleder/tsnep_xdp.c > > diff --git a/drivers/net/ethernet/engleder/Makefile b/drivers/net/ethernet/engleder/Makefile > index b6e3b16623de..0901801cfcc9 100644 > --- a/drivers/net/ethernet/engleder/Makefile > +++ b/drivers/net/ethernet/engleder/Makefile > @@ -6,5 +6,5 @@ > obj-$(CONFIG_TSNEP) += tsnep.o > > tsnep-objs := tsnep_main.o tsnep_ethtool.o tsnep_ptp.o tsnep_tc.o \ > - tsnep_rxnfc.o $(tsnep-y) > + tsnep_rxnfc.o tsnep_xdp.o $(tsnep-y) Not related directly to the subject, but could be fixed in that commit I hope: you don't need to add $(tsnep-y) to $(tsnep-objs), it gets added automatically. > tsnep-$(CONFIG_TSNEP_SELFTESTS) += tsnep_selftests.o > diff --git a/drivers/net/ethernet/engleder/tsnep.h b/drivers/net/ethernet/engleder/tsnep.h > index 29b04127f529..0e7fc36a64e1 100644 > --- a/drivers/net/ethernet/engleder/tsnep.h > +++ b/drivers/net/ethernet/engleder/tsnep.h [...] > @@ -215,6 +220,14 @@ int tsnep_rxnfc_add_rule(struct tsnep_adapter *adapter, > int tsnep_rxnfc_del_rule(struct tsnep_adapter *adapter, > struct ethtool_rxnfc *cmd); > > +int tsnep_xdp_setup_prog(struct tsnep_adapter *adapter, struct bpf_prog *prog, > + struct netlink_ext_ack *extack); > + > +static inline bool tsnep_xdp_is_enabled(struct tsnep_adapter *adapter) > +{ > + return !!adapter->xdp_prog; Any concurrent access protection? READ_ONCE(), RCU? > +} > + > #if IS_ENABLED(CONFIG_TSNEP_SELFTESTS) > int tsnep_ethtool_get_test_count(void); > void tsnep_ethtool_get_test_strings(u8 *data); [...] > +static int tsnep_netdev_bpf(struct net_device *dev, struct netdev_bpf *bpf) > +{ > + struct tsnep_adapter *adapter = netdev_priv(dev); > + > + switch (bpf->command) { > + case XDP_SETUP_PROG: > + return tsnep_xdp_setup_prog(adapter, bpf->prog, bpf->extack); > + default: > + return -EOPNOTSUPP; > + } > +} So, after this commit, I'm able to install an XDP prog to an interface, but it won't do anything. I think the patch could be moved to the end of the series, so that it won't end up with such? > + > static int tsnep_netdev_xdp_xmit(struct net_device *dev, int n, > struct xdp_frame **xdp, u32 flags) > { [...] > +int tsnep_xdp_setup_prog(struct tsnep_adapter *adapter, struct bpf_prog *prog, > + struct netlink_ext_ack *extack) > +{ > + struct net_device *dev = adapter->netdev; > + bool if_running = netif_running(dev); > + struct bpf_prog *old_prog; > + > + if (if_running) > + tsnep_netdev_close(dev); You don't need to close the interface if `!prog == !old_prog`. I wouldn't introduce redundant down-ups here and leave no possibility for prog hotswapping. > + > + old_prog = xchg(&adapter->xdp_prog, prog); > + if (old_prog) > + bpf_prog_put(old_prog); > + > + if (if_running) > + tsnep_netdev_open(dev); > + > + return 0; > +} Thanks, Olek
On 05.01.23 18:24, Alexander Lobakin wrote: > From: Gerhard Engleder <gerhard@engleder-embedded.com> > Date: Wed Jan 04 2023 20:41:29 GMT+0100 > >> Implement setup of BPF programs for XDP RX path with command >> XDP_SETUP_PROG of ndo_bpf(). This is prework for XDP RX path support. >> >> tsnep_netdev_close() is called directly during BPF program setup. Add >> netif_carrier_off() and netif_tx_stop_all_queues() calls to signal to >> network stack that device is down. Otherwise network stack would >> continue transmitting pakets. >> >> Return value of tsnep_netdev_open() is not checked during BPF program >> setup like in other drivers. Forwarding the return value would result in >> a bpf_prog_put() call in dev_xdp_install(), which would make removal of >> BPF program necessary. >> >> If tsnep_netdev_open() fails during BPF program setup, then the network >> stack would call tsnep_netdev_close() anyway. Thus, tsnep_netdev_close() >> checks now if device is already down. >> >> Signed-off-by: Gerhard Engleder <gerhard@engleder-embedded.com> >> --- >> drivers/net/ethernet/engleder/Makefile | 2 +- >> drivers/net/ethernet/engleder/tsnep.h | 13 +++++++++++ >> drivers/net/ethernet/engleder/tsnep_main.c | 25 +++++++++++++++++--- >> drivers/net/ethernet/engleder/tsnep_xdp.c | 27 ++++++++++++++++++++++ >> 4 files changed, 63 insertions(+), 4 deletions(-) >> create mode 100644 drivers/net/ethernet/engleder/tsnep_xdp.c >> >> diff --git a/drivers/net/ethernet/engleder/Makefile b/drivers/net/ethernet/engleder/Makefile >> index b6e3b16623de..0901801cfcc9 100644 >> --- a/drivers/net/ethernet/engleder/Makefile >> +++ b/drivers/net/ethernet/engleder/Makefile >> @@ -6,5 +6,5 @@ >> obj-$(CONFIG_TSNEP) += tsnep.o >> >> tsnep-objs := tsnep_main.o tsnep_ethtool.o tsnep_ptp.o tsnep_tc.o \ >> - tsnep_rxnfc.o $(tsnep-y) >> + tsnep_rxnfc.o tsnep_xdp.o $(tsnep-y) > > Not related directly to the subject, but could be fixed in that commit I > hope: you don't need to add $(tsnep-y) to $(tsnep-objs), it gets added > automatically. I will fix that with this commit. >> tsnep-$(CONFIG_TSNEP_SELFTESTS) += tsnep_selftests.o >> diff --git a/drivers/net/ethernet/engleder/tsnep.h b/drivers/net/ethernet/engleder/tsnep.h >> index 29b04127f529..0e7fc36a64e1 100644 >> --- a/drivers/net/ethernet/engleder/tsnep.h >> +++ b/drivers/net/ethernet/engleder/tsnep.h > > [...] > >> @@ -215,6 +220,14 @@ int tsnep_rxnfc_add_rule(struct tsnep_adapter *adapter, >> int tsnep_rxnfc_del_rule(struct tsnep_adapter *adapter, >> struct ethtool_rxnfc *cmd); >> >> +int tsnep_xdp_setup_prog(struct tsnep_adapter *adapter, struct bpf_prog *prog, >> + struct netlink_ext_ack *extack); >> + >> +static inline bool tsnep_xdp_is_enabled(struct tsnep_adapter *adapter) >> +{ >> + return !!adapter->xdp_prog; > > Any concurrent access protection? READ_ONCE(), RCU? I assume this is about prog hotswapping which you mentioned below. Is concurrent access protection needed without prog hotswapping? >> +} >> + >> #if IS_ENABLED(CONFIG_TSNEP_SELFTESTS) >> int tsnep_ethtool_get_test_count(void); >> void tsnep_ethtool_get_test_strings(u8 *data); > > [...] > >> +static int tsnep_netdev_bpf(struct net_device *dev, struct netdev_bpf *bpf) >> +{ >> + struct tsnep_adapter *adapter = netdev_priv(dev); >> + >> + switch (bpf->command) { >> + case XDP_SETUP_PROG: >> + return tsnep_xdp_setup_prog(adapter, bpf->prog, bpf->extack); >> + default: >> + return -EOPNOTSUPP; >> + } >> +} > > So, after this commit, I'm able to install an XDP prog to an interface, > but it won't do anything. I think the patch could be moved to the end of > the series, so that it won't end up with such? My thinking was to first implement the base and the actual functionality last. I can move it to the end. >> + >> static int tsnep_netdev_xdp_xmit(struct net_device *dev, int n, >> struct xdp_frame **xdp, u32 flags) >> { > > [...] > >> +int tsnep_xdp_setup_prog(struct tsnep_adapter *adapter, struct bpf_prog *prog, >> + struct netlink_ext_ack *extack) >> +{ >> + struct net_device *dev = adapter->netdev; >> + bool if_running = netif_running(dev); >> + struct bpf_prog *old_prog; >> + >> + if (if_running) >> + tsnep_netdev_close(dev); > > You don't need to close the interface if `!prog == !old_prog`. I > wouldn't introduce redundant down-ups here and leave no possibility for > prog hotswapping. If prog hotswapping is possible, then how shall it be ensured that the old prog is not running anymore before 'bpf_prog_put(old_prog)' is called? I don't understand how 'READ_ONCE(adapter->xdp_prog)' during RX path and 'old_prog = xchg(&adapter->xdp_prog, prog)' during prog setup can ensure that (e.g. in ixgbe driver). Gerhard
diff --git a/drivers/net/ethernet/engleder/Makefile b/drivers/net/ethernet/engleder/Makefile index b6e3b16623de..0901801cfcc9 100644 --- a/drivers/net/ethernet/engleder/Makefile +++ b/drivers/net/ethernet/engleder/Makefile @@ -6,5 +6,5 @@ obj-$(CONFIG_TSNEP) += tsnep.o tsnep-objs := tsnep_main.o tsnep_ethtool.o tsnep_ptp.o tsnep_tc.o \ - tsnep_rxnfc.o $(tsnep-y) + tsnep_rxnfc.o tsnep_xdp.o $(tsnep-y) tsnep-$(CONFIG_TSNEP_SELFTESTS) += tsnep_selftests.o diff --git a/drivers/net/ethernet/engleder/tsnep.h b/drivers/net/ethernet/engleder/tsnep.h index 29b04127f529..0e7fc36a64e1 100644 --- a/drivers/net/ethernet/engleder/tsnep.h +++ b/drivers/net/ethernet/engleder/tsnep.h @@ -183,6 +183,8 @@ struct tsnep_adapter { int rxnfc_count; int rxnfc_max; + struct bpf_prog *xdp_prog; + int num_tx_queues; struct tsnep_tx tx[TSNEP_MAX_QUEUES]; int num_rx_queues; @@ -192,6 +194,9 @@ struct tsnep_adapter { struct tsnep_queue queue[TSNEP_MAX_QUEUES]; }; +int tsnep_netdev_open(struct net_device *netdev); +int tsnep_netdev_close(struct net_device *netdev); + extern const struct ethtool_ops tsnep_ethtool_ops; int tsnep_ptp_init(struct tsnep_adapter *adapter); @@ -215,6 +220,14 @@ int tsnep_rxnfc_add_rule(struct tsnep_adapter *adapter, int tsnep_rxnfc_del_rule(struct tsnep_adapter *adapter, struct ethtool_rxnfc *cmd); +int tsnep_xdp_setup_prog(struct tsnep_adapter *adapter, struct bpf_prog *prog, + struct netlink_ext_ack *extack); + +static inline bool tsnep_xdp_is_enabled(struct tsnep_adapter *adapter) +{ + return !!adapter->xdp_prog; +} + #if IS_ENABLED(CONFIG_TSNEP_SELFTESTS) int tsnep_ethtool_get_test_count(void); void tsnep_ethtool_get_test_strings(u8 *data); diff --git a/drivers/net/ethernet/engleder/tsnep_main.c b/drivers/net/ethernet/engleder/tsnep_main.c index 9a666dbbe8cd..a603b79b7411 100644 --- a/drivers/net/ethernet/engleder/tsnep_main.c +++ b/drivers/net/ethernet/engleder/tsnep_main.c @@ -1236,7 +1236,7 @@ static void tsnep_free_irq(struct tsnep_queue *queue, bool first) memset(queue->name, 0, sizeof(queue->name)); } -static int tsnep_netdev_open(struct net_device *netdev) +int tsnep_netdev_open(struct net_device *netdev) { struct tsnep_adapter *adapter = netdev_priv(netdev); int i; @@ -1296,6 +1296,8 @@ static int tsnep_netdev_open(struct net_device *netdev) tsnep_enable_irq(adapter, adapter->queue[i].irq_mask); } + netif_tx_start_all_queues(adapter->netdev); + clear_bit(__TSNEP_DOWN, &adapter->state); return 0; @@ -1315,12 +1317,16 @@ static int tsnep_netdev_open(struct net_device *netdev) return retval; } -static int tsnep_netdev_close(struct net_device *netdev) +int tsnep_netdev_close(struct net_device *netdev) { struct tsnep_adapter *adapter = netdev_priv(netdev); int i; - set_bit(__TSNEP_DOWN, &adapter->state); + if (test_and_set_bit(__TSNEP_DOWN, &adapter->state)) + return 0; + + netif_carrier_off(netdev); + netif_tx_stop_all_queues(netdev); tsnep_disable_irq(adapter, ECM_INT_LINK); tsnep_phy_close(adapter); @@ -1484,6 +1490,18 @@ static ktime_t tsnep_netdev_get_tstamp(struct net_device *netdev, return ns_to_ktime(timestamp); } +static int tsnep_netdev_bpf(struct net_device *dev, struct netdev_bpf *bpf) +{ + struct tsnep_adapter *adapter = netdev_priv(dev); + + switch (bpf->command) { + case XDP_SETUP_PROG: + return tsnep_xdp_setup_prog(adapter, bpf->prog, bpf->extack); + default: + return -EOPNOTSUPP; + } +} + static int tsnep_netdev_xdp_xmit(struct net_device *dev, int n, struct xdp_frame **xdp, u32 flags) { @@ -1535,6 +1553,7 @@ static const struct net_device_ops tsnep_netdev_ops = { .ndo_set_features = tsnep_netdev_set_features, .ndo_get_tstamp = tsnep_netdev_get_tstamp, .ndo_setup_tc = tsnep_tc_setup, + .ndo_bpf = tsnep_netdev_bpf, .ndo_xdp_xmit = tsnep_netdev_xdp_xmit, }; diff --git a/drivers/net/ethernet/engleder/tsnep_xdp.c b/drivers/net/ethernet/engleder/tsnep_xdp.c new file mode 100644 index 000000000000..02d84dfbdde4 --- /dev/null +++ b/drivers/net/ethernet/engleder/tsnep_xdp.c @@ -0,0 +1,27 @@ +// SPDX-License-Identifier: GPL-2.0 +/* Copyright (C) 2022 Gerhard Engleder <gerhard@engleder-embedded.com> */ + +#include <linux/if_vlan.h> +#include <net/xdp_sock_drv.h> + +#include "tsnep.h" + +int tsnep_xdp_setup_prog(struct tsnep_adapter *adapter, struct bpf_prog *prog, + struct netlink_ext_ack *extack) +{ + struct net_device *dev = adapter->netdev; + bool if_running = netif_running(dev); + struct bpf_prog *old_prog; + + if (if_running) + tsnep_netdev_close(dev); + + old_prog = xchg(&adapter->xdp_prog, prog); + if (old_prog) + bpf_prog_put(old_prog); + + if (if_running) + tsnep_netdev_open(dev); + + return 0; +}
Implement setup of BPF programs for XDP RX path with command XDP_SETUP_PROG of ndo_bpf(). This is prework for XDP RX path support. tsnep_netdev_close() is called directly during BPF program setup. Add netif_carrier_off() and netif_tx_stop_all_queues() calls to signal to network stack that device is down. Otherwise network stack would continue transmitting pakets. Return value of tsnep_netdev_open() is not checked during BPF program setup like in other drivers. Forwarding the return value would result in a bpf_prog_put() call in dev_xdp_install(), which would make removal of BPF program necessary. If tsnep_netdev_open() fails during BPF program setup, then the network stack would call tsnep_netdev_close() anyway. Thus, tsnep_netdev_close() checks now if device is already down. Signed-off-by: Gerhard Engleder <gerhard@engleder-embedded.com> --- drivers/net/ethernet/engleder/Makefile | 2 +- drivers/net/ethernet/engleder/tsnep.h | 13 +++++++++++ drivers/net/ethernet/engleder/tsnep_main.c | 25 +++++++++++++++++--- drivers/net/ethernet/engleder/tsnep_xdp.c | 27 ++++++++++++++++++++++ 4 files changed, 63 insertions(+), 4 deletions(-) create mode 100644 drivers/net/ethernet/engleder/tsnep_xdp.c