forked from mirrors/linux
		
	net: sched: introduce terse dump flag
Add new TCA_DUMP_FLAGS attribute and use it in cls API to request terse filter output from classifiers with TCA_DUMP_FLAGS_TERSE flag. This option is intended to be used to improve performance of TC filter dump when userland only needs to obtain stats and not the whole classifier/action data. Extend struct tcf_proto_ops with new terse_dump() callback that must be defined by supporting classifier implementations. Support of the options in specific classifiers and actions is implemented in following patches in the series. Signed-off-by: Vlad Buslov <vladbu@mellanox.com> Reviewed-by: Jiri Pirko <jiri@mellanox.com> Signed-off-by: David S. Miller <davem@davemloft.net>
This commit is contained in:
		
							parent
							
								
									2e186a2cf8
								
							
						
					
					
						commit
						f8ab1807a9
					
				
					 3 changed files with 41 additions and 8 deletions
				
			
		| 
						 | 
					@ -330,6 +330,10 @@ struct tcf_proto_ops {
 | 
				
			||||||
	int			(*dump)(struct net*, struct tcf_proto*, void *,
 | 
						int			(*dump)(struct net*, struct tcf_proto*, void *,
 | 
				
			||||||
					struct sk_buff *skb, struct tcmsg*,
 | 
										struct sk_buff *skb, struct tcmsg*,
 | 
				
			||||||
					bool);
 | 
										bool);
 | 
				
			||||||
 | 
						int			(*terse_dump)(struct net *net,
 | 
				
			||||||
 | 
										      struct tcf_proto *tp, void *fh,
 | 
				
			||||||
 | 
										      struct sk_buff *skb,
 | 
				
			||||||
 | 
										      struct tcmsg *t, bool rtnl_held);
 | 
				
			||||||
	int			(*tmplt_dump)(struct sk_buff *skb,
 | 
						int			(*tmplt_dump)(struct sk_buff *skb,
 | 
				
			||||||
					      struct net *net,
 | 
										      struct net *net,
 | 
				
			||||||
					      void *tmplt_priv);
 | 
										      void *tmplt_priv);
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -609,11 +609,17 @@ enum {
 | 
				
			||||||
	TCA_HW_OFFLOAD,
 | 
						TCA_HW_OFFLOAD,
 | 
				
			||||||
	TCA_INGRESS_BLOCK,
 | 
						TCA_INGRESS_BLOCK,
 | 
				
			||||||
	TCA_EGRESS_BLOCK,
 | 
						TCA_EGRESS_BLOCK,
 | 
				
			||||||
 | 
						TCA_DUMP_FLAGS,
 | 
				
			||||||
	__TCA_MAX
 | 
						__TCA_MAX
 | 
				
			||||||
};
 | 
					};
 | 
				
			||||||
 | 
					
 | 
				
			||||||
#define TCA_MAX (__TCA_MAX - 1)
 | 
					#define TCA_MAX (__TCA_MAX - 1)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					#define TCA_DUMP_FLAGS_TERSE (1 << 0) /* Means that in dump user gets only basic
 | 
				
			||||||
 | 
									       * data necessary to identify the objects
 | 
				
			||||||
 | 
									       * (handle, cookie, etc.) and stats.
 | 
				
			||||||
 | 
									       */
 | 
				
			||||||
 | 
					
 | 
				
			||||||
#define TCA_RTA(r)  ((struct rtattr*)(((char*)(r)) + NLMSG_ALIGN(sizeof(struct tcmsg))))
 | 
					#define TCA_RTA(r)  ((struct rtattr*)(((char*)(r)) + NLMSG_ALIGN(sizeof(struct tcmsg))))
 | 
				
			||||||
#define TCA_PAYLOAD(n) NLMSG_PAYLOAD(n,sizeof(struct tcmsg))
 | 
					#define TCA_PAYLOAD(n) NLMSG_PAYLOAD(n,sizeof(struct tcmsg))
 | 
				
			||||||
 | 
					
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -1851,7 +1851,7 @@ static int tcf_fill_node(struct net *net, struct sk_buff *skb,
 | 
				
			||||||
			 struct tcf_proto *tp, struct tcf_block *block,
 | 
								 struct tcf_proto *tp, struct tcf_block *block,
 | 
				
			||||||
			 struct Qdisc *q, u32 parent, void *fh,
 | 
								 struct Qdisc *q, u32 parent, void *fh,
 | 
				
			||||||
			 u32 portid, u32 seq, u16 flags, int event,
 | 
								 u32 portid, u32 seq, u16 flags, int event,
 | 
				
			||||||
			 bool rtnl_held)
 | 
								 bool terse_dump, bool rtnl_held)
 | 
				
			||||||
{
 | 
					{
 | 
				
			||||||
	struct tcmsg *tcm;
 | 
						struct tcmsg *tcm;
 | 
				
			||||||
	struct nlmsghdr  *nlh;
 | 
						struct nlmsghdr  *nlh;
 | 
				
			||||||
| 
						 | 
					@ -1878,6 +1878,14 @@ static int tcf_fill_node(struct net *net, struct sk_buff *skb,
 | 
				
			||||||
		goto nla_put_failure;
 | 
							goto nla_put_failure;
 | 
				
			||||||
	if (!fh) {
 | 
						if (!fh) {
 | 
				
			||||||
		tcm->tcm_handle = 0;
 | 
							tcm->tcm_handle = 0;
 | 
				
			||||||
 | 
						} else if (terse_dump) {
 | 
				
			||||||
 | 
							if (tp->ops->terse_dump) {
 | 
				
			||||||
 | 
								if (tp->ops->terse_dump(net, tp, fh, skb, tcm,
 | 
				
			||||||
 | 
											rtnl_held) < 0)
 | 
				
			||||||
 | 
									goto nla_put_failure;
 | 
				
			||||||
 | 
							} else {
 | 
				
			||||||
 | 
								goto cls_op_not_supp;
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
	} else {
 | 
						} else {
 | 
				
			||||||
		if (tp->ops->dump &&
 | 
							if (tp->ops->dump &&
 | 
				
			||||||
		    tp->ops->dump(net, tp, fh, skb, tcm, rtnl_held) < 0)
 | 
							    tp->ops->dump(net, tp, fh, skb, tcm, rtnl_held) < 0)
 | 
				
			||||||
| 
						 | 
					@ -1888,6 +1896,7 @@ static int tcf_fill_node(struct net *net, struct sk_buff *skb,
 | 
				
			||||||
 | 
					
 | 
				
			||||||
out_nlmsg_trim:
 | 
					out_nlmsg_trim:
 | 
				
			||||||
nla_put_failure:
 | 
					nla_put_failure:
 | 
				
			||||||
 | 
					cls_op_not_supp:
 | 
				
			||||||
	nlmsg_trim(skb, b);
 | 
						nlmsg_trim(skb, b);
 | 
				
			||||||
	return -1;
 | 
						return -1;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
| 
						 | 
					@ -1908,7 +1917,7 @@ static int tfilter_notify(struct net *net, struct sk_buff *oskb,
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	if (tcf_fill_node(net, skb, tp, block, q, parent, fh, portid,
 | 
						if (tcf_fill_node(net, skb, tp, block, q, parent, fh, portid,
 | 
				
			||||||
			  n->nlmsg_seq, n->nlmsg_flags, event,
 | 
								  n->nlmsg_seq, n->nlmsg_flags, event,
 | 
				
			||||||
			  rtnl_held) <= 0) {
 | 
								  false, rtnl_held) <= 0) {
 | 
				
			||||||
		kfree_skb(skb);
 | 
							kfree_skb(skb);
 | 
				
			||||||
		return -EINVAL;
 | 
							return -EINVAL;
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
| 
						 | 
					@ -1940,7 +1949,7 @@ static int tfilter_del_notify(struct net *net, struct sk_buff *oskb,
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	if (tcf_fill_node(net, skb, tp, block, q, parent, fh, portid,
 | 
						if (tcf_fill_node(net, skb, tp, block, q, parent, fh, portid,
 | 
				
			||||||
			  n->nlmsg_seq, n->nlmsg_flags, RTM_DELTFILTER,
 | 
								  n->nlmsg_seq, n->nlmsg_flags, RTM_DELTFILTER,
 | 
				
			||||||
			  rtnl_held) <= 0) {
 | 
								  false, rtnl_held) <= 0) {
 | 
				
			||||||
		NL_SET_ERR_MSG(extack, "Failed to build del event notification");
 | 
							NL_SET_ERR_MSG(extack, "Failed to build del event notification");
 | 
				
			||||||
		kfree_skb(skb);
 | 
							kfree_skb(skb);
 | 
				
			||||||
		return -EINVAL;
 | 
							return -EINVAL;
 | 
				
			||||||
| 
						 | 
					@ -2501,6 +2510,7 @@ struct tcf_dump_args {
 | 
				
			||||||
	struct tcf_block *block;
 | 
						struct tcf_block *block;
 | 
				
			||||||
	struct Qdisc *q;
 | 
						struct Qdisc *q;
 | 
				
			||||||
	u32 parent;
 | 
						u32 parent;
 | 
				
			||||||
 | 
						bool terse_dump;
 | 
				
			||||||
};
 | 
					};
 | 
				
			||||||
 | 
					
 | 
				
			||||||
static int tcf_node_dump(struct tcf_proto *tp, void *n, struct tcf_walker *arg)
 | 
					static int tcf_node_dump(struct tcf_proto *tp, void *n, struct tcf_walker *arg)
 | 
				
			||||||
| 
						 | 
					@ -2511,12 +2521,12 @@ static int tcf_node_dump(struct tcf_proto *tp, void *n, struct tcf_walker *arg)
 | 
				
			||||||
	return tcf_fill_node(net, a->skb, tp, a->block, a->q, a->parent,
 | 
						return tcf_fill_node(net, a->skb, tp, a->block, a->q, a->parent,
 | 
				
			||||||
			     n, NETLINK_CB(a->cb->skb).portid,
 | 
								     n, NETLINK_CB(a->cb->skb).portid,
 | 
				
			||||||
			     a->cb->nlh->nlmsg_seq, NLM_F_MULTI,
 | 
								     a->cb->nlh->nlmsg_seq, NLM_F_MULTI,
 | 
				
			||||||
			     RTM_NEWTFILTER, true);
 | 
								     RTM_NEWTFILTER, a->terse_dump, true);
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
static bool tcf_chain_dump(struct tcf_chain *chain, struct Qdisc *q, u32 parent,
 | 
					static bool tcf_chain_dump(struct tcf_chain *chain, struct Qdisc *q, u32 parent,
 | 
				
			||||||
			   struct sk_buff *skb, struct netlink_callback *cb,
 | 
								   struct sk_buff *skb, struct netlink_callback *cb,
 | 
				
			||||||
			   long index_start, long *p_index)
 | 
								   long index_start, long *p_index, bool terse)
 | 
				
			||||||
{
 | 
					{
 | 
				
			||||||
	struct net *net = sock_net(skb->sk);
 | 
						struct net *net = sock_net(skb->sk);
 | 
				
			||||||
	struct tcf_block *block = chain->block;
 | 
						struct tcf_block *block = chain->block;
 | 
				
			||||||
| 
						 | 
					@ -2545,7 +2555,7 @@ static bool tcf_chain_dump(struct tcf_chain *chain, struct Qdisc *q, u32 parent,
 | 
				
			||||||
			if (tcf_fill_node(net, skb, tp, block, q, parent, NULL,
 | 
								if (tcf_fill_node(net, skb, tp, block, q, parent, NULL,
 | 
				
			||||||
					  NETLINK_CB(cb->skb).portid,
 | 
										  NETLINK_CB(cb->skb).portid,
 | 
				
			||||||
					  cb->nlh->nlmsg_seq, NLM_F_MULTI,
 | 
										  cb->nlh->nlmsg_seq, NLM_F_MULTI,
 | 
				
			||||||
					  RTM_NEWTFILTER, true) <= 0)
 | 
										  RTM_NEWTFILTER, false, true) <= 0)
 | 
				
			||||||
				goto errout;
 | 
									goto errout;
 | 
				
			||||||
			cb->args[1] = 1;
 | 
								cb->args[1] = 1;
 | 
				
			||||||
		}
 | 
							}
 | 
				
			||||||
| 
						 | 
					@ -2561,6 +2571,7 @@ static bool tcf_chain_dump(struct tcf_chain *chain, struct Qdisc *q, u32 parent,
 | 
				
			||||||
		arg.w.skip = cb->args[1] - 1;
 | 
							arg.w.skip = cb->args[1] - 1;
 | 
				
			||||||
		arg.w.count = 0;
 | 
							arg.w.count = 0;
 | 
				
			||||||
		arg.w.cookie = cb->args[2];
 | 
							arg.w.cookie = cb->args[2];
 | 
				
			||||||
 | 
							arg.terse_dump = terse;
 | 
				
			||||||
		tp->ops->walk(tp, &arg.w, true);
 | 
							tp->ops->walk(tp, &arg.w, true);
 | 
				
			||||||
		cb->args[2] = arg.w.cookie;
 | 
							cb->args[2] = arg.w.cookie;
 | 
				
			||||||
		cb->args[1] = arg.w.count + 1;
 | 
							cb->args[1] = arg.w.count + 1;
 | 
				
			||||||
| 
						 | 
					@ -2574,6 +2585,10 @@ static bool tcf_chain_dump(struct tcf_chain *chain, struct Qdisc *q, u32 parent,
 | 
				
			||||||
	return false;
 | 
						return false;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					static const struct nla_policy tcf_tfilter_dump_policy[TCA_MAX + 1] = {
 | 
				
			||||||
 | 
						[TCA_DUMP_FLAGS] = NLA_POLICY_BITFIELD32(TCA_DUMP_FLAGS_TERSE),
 | 
				
			||||||
 | 
					};
 | 
				
			||||||
 | 
					
 | 
				
			||||||
/* called with RTNL */
 | 
					/* called with RTNL */
 | 
				
			||||||
static int tc_dump_tfilter(struct sk_buff *skb, struct netlink_callback *cb)
 | 
					static int tc_dump_tfilter(struct sk_buff *skb, struct netlink_callback *cb)
 | 
				
			||||||
{
 | 
					{
 | 
				
			||||||
| 
						 | 
					@ -2583,6 +2598,7 @@ static int tc_dump_tfilter(struct sk_buff *skb, struct netlink_callback *cb)
 | 
				
			||||||
	struct Qdisc *q = NULL;
 | 
						struct Qdisc *q = NULL;
 | 
				
			||||||
	struct tcf_block *block;
 | 
						struct tcf_block *block;
 | 
				
			||||||
	struct tcmsg *tcm = nlmsg_data(cb->nlh);
 | 
						struct tcmsg *tcm = nlmsg_data(cb->nlh);
 | 
				
			||||||
 | 
						bool terse_dump = false;
 | 
				
			||||||
	long index_start;
 | 
						long index_start;
 | 
				
			||||||
	long index;
 | 
						long index;
 | 
				
			||||||
	u32 parent;
 | 
						u32 parent;
 | 
				
			||||||
| 
						 | 
					@ -2592,10 +2608,17 @@ static int tc_dump_tfilter(struct sk_buff *skb, struct netlink_callback *cb)
 | 
				
			||||||
		return skb->len;
 | 
							return skb->len;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	err = nlmsg_parse_deprecated(cb->nlh, sizeof(*tcm), tca, TCA_MAX,
 | 
						err = nlmsg_parse_deprecated(cb->nlh, sizeof(*tcm), tca, TCA_MAX,
 | 
				
			||||||
				     NULL, cb->extack);
 | 
									     tcf_tfilter_dump_policy, cb->extack);
 | 
				
			||||||
	if (err)
 | 
						if (err)
 | 
				
			||||||
		return err;
 | 
							return err;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						if (tca[TCA_DUMP_FLAGS]) {
 | 
				
			||||||
 | 
							struct nla_bitfield32 flags =
 | 
				
			||||||
 | 
								nla_get_bitfield32(tca[TCA_DUMP_FLAGS]);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							terse_dump = flags.value & TCA_DUMP_FLAGS_TERSE;
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	if (tcm->tcm_ifindex == TCM_IFINDEX_MAGIC_BLOCK) {
 | 
						if (tcm->tcm_ifindex == TCM_IFINDEX_MAGIC_BLOCK) {
 | 
				
			||||||
		block = tcf_block_refcnt_get(net, tcm->tcm_block_index);
 | 
							block = tcf_block_refcnt_get(net, tcm->tcm_block_index);
 | 
				
			||||||
		if (!block)
 | 
							if (!block)
 | 
				
			||||||
| 
						 | 
					@ -2653,7 +2676,7 @@ static int tc_dump_tfilter(struct sk_buff *skb, struct netlink_callback *cb)
 | 
				
			||||||
		    nla_get_u32(tca[TCA_CHAIN]) != chain->index)
 | 
							    nla_get_u32(tca[TCA_CHAIN]) != chain->index)
 | 
				
			||||||
			continue;
 | 
								continue;
 | 
				
			||||||
		if (!tcf_chain_dump(chain, q, parent, skb, cb,
 | 
							if (!tcf_chain_dump(chain, q, parent, skb, cb,
 | 
				
			||||||
				    index_start, &index)) {
 | 
									    index_start, &index, terse_dump)) {
 | 
				
			||||||
			tcf_chain_put(chain);
 | 
								tcf_chain_put(chain);
 | 
				
			||||||
			err = -EMSGSIZE;
 | 
								err = -EMSGSIZE;
 | 
				
			||||||
			break;
 | 
								break;
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
		Loading…
	
		Reference in a new issue