#define NH_DEV_HASHBITS  8
 #define NH_DEV_HASHSIZE (1U << NH_DEV_HASHBITS)
 
+#define NHA_OP_FLAGS_DUMP_ALL (NHA_OP_FLAG_DUMP_STATS)
+
 static const struct nla_policy rtm_nh_policy_new[] = {
        [NHA_ID]                = { .type = NLA_U32 },
        [NHA_GROUP]             = { .type = NLA_BINARY },
 
 static const struct nla_policy rtm_nh_policy_get[] = {
        [NHA_ID]                = { .type = NLA_U32 },
-       [NHA_OP_FLAGS]          = NLA_POLICY_MASK(NLA_U32, 0),
+       [NHA_OP_FLAGS]          = NLA_POLICY_MASK(NLA_U32,
+                                                 NHA_OP_FLAGS_DUMP_ALL),
 };
 
 static const struct nla_policy rtm_nh_policy_del[] = {
        [NHA_GROUPS]            = { .type = NLA_FLAG },
        [NHA_MASTER]            = { .type = NLA_U32 },
        [NHA_FDB]               = { .type = NLA_FLAG },
-       [NHA_OP_FLAGS]          = NLA_POLICY_MASK(NLA_U32, 0),
+       [NHA_OP_FLAGS]          = NLA_POLICY_MASK(NLA_U32,
+                                                 NHA_OP_FLAGS_DUMP_ALL),
 };
 
 static const struct nla_policy rtm_nh_res_policy_new[] = {
        u64_stats_update_end(&cpu_stats->syncp);
 }
 
-static int nla_put_nh_group(struct sk_buff *skb, struct nh_group *nhg)
+static void nh_grp_entry_stats_read(struct nh_grp_entry *nhge,
+                                   u64 *ret_packets)
+{
+       int i;
+
+       *ret_packets = 0;
+
+       for_each_possible_cpu(i) {
+               struct nh_grp_entry_stats *cpu_stats;
+               unsigned int start;
+               u64 packets;
+
+               cpu_stats = per_cpu_ptr(nhge->stats, i);
+               do {
+                       start = u64_stats_fetch_begin(&cpu_stats->syncp);
+                       packets = u64_stats_read(&cpu_stats->packets);
+               } while (u64_stats_fetch_retry(&cpu_stats->syncp, start));
+
+               *ret_packets += packets;
+       }
+}
+
+static int nla_put_nh_group_stats_entry(struct sk_buff *skb,
+                                       struct nh_grp_entry *nhge)
+{
+       struct nlattr *nest;
+       u64 packets;
+
+       nh_grp_entry_stats_read(nhge, &packets);
+
+       nest = nla_nest_start(skb, NHA_GROUP_STATS_ENTRY);
+       if (!nest)
+               return -EMSGSIZE;
+
+       if (nla_put_u32(skb, NHA_GROUP_STATS_ENTRY_ID, nhge->nh->id) ||
+           nla_put_uint(skb, NHA_GROUP_STATS_ENTRY_PACKETS, packets))
+               goto nla_put_failure;
+
+       nla_nest_end(skb, nest);
+       return 0;
+
+nla_put_failure:
+       nla_nest_cancel(skb, nest);
+       return -EMSGSIZE;
+}
+
+static int nla_put_nh_group_stats(struct sk_buff *skb, struct nexthop *nh)
 {
+       struct nh_group *nhg = rtnl_dereference(nh->nh_grp);
+       struct nlattr *nest;
+       int i;
+
+       nest = nla_nest_start(skb, NHA_GROUP_STATS);
+       if (!nest)
+               return -EMSGSIZE;
+
+       for (i = 0; i < nhg->num_nh; i++)
+               if (nla_put_nh_group_stats_entry(skb, &nhg->nh_entries[i]))
+                       goto cancel_out;
+
+       nla_nest_end(skb, nest);
+       return 0;
+
+cancel_out:
+       nla_nest_cancel(skb, nest);
+       return -EMSGSIZE;
+}
+
+static int nla_put_nh_group(struct sk_buff *skb, struct nexthop *nh,
+                           u32 op_flags)
+{
+       struct nh_group *nhg = rtnl_dereference(nh->nh_grp);
        struct nexthop_grp *p;
        size_t len = nhg->num_nh * sizeof(*p);
        struct nlattr *nla;
        if (nhg->resilient && nla_put_nh_group_res(skb, nhg))
                goto nla_put_failure;
 
+       if (op_flags & NHA_OP_FLAG_DUMP_STATS &&
+           nla_put_nh_group_stats(skb, nh))
+               goto nla_put_failure;
+
        return 0;
 
 nla_put_failure:
 }
 
 static int nh_fill_node(struct sk_buff *skb, struct nexthop *nh,
-                       int event, u32 portid, u32 seq, unsigned int nlflags)
+                       int event, u32 portid, u32 seq, unsigned int nlflags,
+                       u32 op_flags)
 {
        struct fib6_nh *fib6_nh;
        struct fib_nh *fib_nh;
 
                if (nhg->fdb_nh && nla_put_flag(skb, NHA_FDB))
                        goto nla_put_failure;
-               if (nla_put_nh_group(skb, nhg))
+               if (nla_put_nh_group(skb, nh, op_flags))
                        goto nla_put_failure;
                goto out;
        }
        if (!skb)
                goto errout;
 
-       err = nh_fill_node(skb, nh, event, info->portid, seq, nlflags);
+       err = nh_fill_node(skb, nh, event, info->portid, seq, nlflags, 0);
        if (err < 0) {
                /* -EMSGSIZE implies BUG in nh_nlmsg_size() */
                WARN_ON(err == -EMSGSIZE);
                goto errout_free;
 
        err = nh_fill_node(skb, nh, RTM_NEWNEXTHOP, NETLINK_CB(in_skb).portid,
-                          nlh->nlmsg_seq, 0);
+                          nlh->nlmsg_seq, 0, op_flags);
        if (err < 0) {
                WARN_ON(err == -EMSGSIZE);
                goto errout_free;
 
        return nh_fill_node(skb, nh, RTM_NEWNEXTHOP,
                            NETLINK_CB(cb->skb).portid,
-                           cb->nlh->nlmsg_seq, NLM_F_MULTI);
+                           cb->nlh->nlmsg_seq, NLM_F_MULTI, filter->op_flags);
 }
 
 /* rtnl */