xfrm: get global statistics from the offloaded device
authorLeon Romanovsky <leonro@nvidia.com>
Wed, 4 Oct 2023 11:11:48 +0000 (14:11 +0300)
committerSaeed Mahameed <saeedm@nvidia.com>
Tue, 6 Feb 2024 00:45:49 +0000 (16:45 -0800)
Iterate over all SAs in order to fill global IPsec statistics.

Acked-by: Steffen Klassert <steffen.klassert@secunet.com>
Signed-off-by: Leon Romanovsky <leonro@nvidia.com>
Signed-off-by: Saeed Mahameed <saeedm@nvidia.com>
drivers/net/ethernet/mellanox/mlx5/core/en_accel/ipsec.c
include/net/xfrm.h
net/xfrm/xfrm_proc.c
net/xfrm/xfrm_state.c

index f160522fbe7573c2f78491d4d1b372f78946607f..46cce4e38d8463b734ae220b096227f951e3d041 100644 (file)
@@ -991,7 +991,8 @@ static void mlx5e_xfrm_update_stats(struct xfrm_state *x)
        u64 packets, bytes, lastuse;
 
        lockdep_assert(lockdep_is_held(&x->lock) ||
-                      lockdep_is_held(&dev_net(x->xso.real_dev)->xfrm.xfrm_cfg_mutex));
+                      lockdep_is_held(&dev_net(x->xso.real_dev)->xfrm.xfrm_cfg_mutex) ||
+                      lockdep_is_held(&dev_net(x->xso.real_dev)->xfrm.xfrm_state_lock));
 
        if (x->xso.flags & XFRM_DEV_OFFLOAD_FLAG_ACQ ||
            x->xso.type != XFRM_DEV_OFFLOAD_PACKET)
index 4ca2f3205190c94d70a1eb10c9cf947f778d6d43..57c743b7e4fef1146ff49d23b38f54c0c7b99c1a 100644 (file)
 
 #ifdef CONFIG_XFRM_STATISTICS
 #define XFRM_INC_STATS(net, field)     SNMP_INC_STATS((net)->mib.xfrm_statistics, field)
+#define XFRM_ADD_STATS(net, field, val) SNMP_ADD_STATS((net)->mib.xfrm_statistics, field, val)
 #else
 #define XFRM_INC_STATS(net, field)     ((void)(net))
+#define XFRM_ADD_STATS(net, field, val) ((void)(net))
 #endif
 
 
@@ -1577,6 +1579,7 @@ struct xfrm_state *xfrm_stateonly_find(struct net *net, u32 mark, u32 if_id,
 struct xfrm_state *xfrm_state_lookup_byspi(struct net *net, __be32 spi,
                                              unsigned short family);
 int xfrm_state_check_expire(struct xfrm_state *x);
+void xfrm_state_update_stats(struct net *net);
 #ifdef CONFIG_XFRM_OFFLOAD
 static inline void xfrm_dev_state_update_stats(struct xfrm_state *x)
 {
index fee9b5cf37a7c22be761c61ba133d03281c116e7..5f9bf8e5c9338f4337b308f06edb25612c68a3fa 100644 (file)
@@ -52,6 +52,7 @@ static int xfrm_statistics_seq_show(struct seq_file *seq, void *v)
 
        memset(buff, 0, sizeof(unsigned long) * LINUX_MIB_XFRMMAX);
 
+       xfrm_state_update_stats(net);
        snmp_get_cpu_field_batch(buff, xfrm_mib_list,
                                 net->mib.xfrm_statistics);
        for (i = 0; xfrm_mib_list[i].name; i++)
index d8701b2d0d57b9474292c450ba0edd7f3aa5a276..0c306473a79d13321ae8ab93b97fe717c00cdab3 100644 (file)
@@ -1957,6 +1957,19 @@ int xfrm_state_check_expire(struct xfrm_state *x)
 }
 EXPORT_SYMBOL(xfrm_state_check_expire);
 
+void xfrm_state_update_stats(struct net *net)
+{
+       struct xfrm_state *x;
+       int i;
+
+       spin_lock_bh(&net->xfrm.xfrm_state_lock);
+       for (i = 0; i <= net->xfrm.state_hmask; i++) {
+               hlist_for_each_entry(x, net->xfrm.state_bydst + i, bydst)
+                       xfrm_dev_state_update_stats(x);
+       }
+       spin_unlock_bh(&net->xfrm.xfrm_state_lock);
+}
+
 struct xfrm_state *
 xfrm_state_lookup(struct net *net, u32 mark, const xfrm_address_t *daddr, __be32 spi,
                  u8 proto, unsigned short family)