struct tcf_proto *tp, struct tcf_block *block,
                         struct Qdisc *q, u32 parent, void *fh,
                         u32 portid, u32 seq, u16 flags, int event,
-                        bool rtnl_held)
+                        bool terse_dump, bool rtnl_held)
 {
        struct tcmsg *tcm;
        struct nlmsghdr  *nlh;
                goto nla_put_failure;
        if (!fh) {
                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 {
                if (tp->ops->dump &&
                    tp->ops->dump(net, tp, fh, skb, tcm, rtnl_held) < 0)
 
 out_nlmsg_trim:
 nla_put_failure:
+cls_op_not_supp:
        nlmsg_trim(skb, b);
        return -1;
 }
 
        if (tcf_fill_node(net, skb, tp, block, q, parent, fh, portid,
                          n->nlmsg_seq, n->nlmsg_flags, event,
-                         rtnl_held) <= 0) {
+                         false, rtnl_held) <= 0) {
                kfree_skb(skb);
                return -EINVAL;
        }
 
        if (tcf_fill_node(net, skb, tp, block, q, parent, fh, portid,
                          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");
                kfree_skb(skb);
                return -EINVAL;
        struct tcf_block *block;
        struct Qdisc *q;
        u32 parent;
+       bool terse_dump;
 };
 
 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,
                             n, NETLINK_CB(a->cb->skb).portid,
                             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,
                           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 tcf_block *block = chain->block;
                        if (tcf_fill_node(net, skb, tp, block, q, parent, NULL,
                                          NETLINK_CB(cb->skb).portid,
                                          cb->nlh->nlmsg_seq, NLM_F_MULTI,
-                                         RTM_NEWTFILTER, true) <= 0)
+                                         RTM_NEWTFILTER, false, true) <= 0)
                                goto errout;
                        cb->args[1] = 1;
                }
                arg.w.skip = cb->args[1] - 1;
                arg.w.count = 0;
                arg.w.cookie = cb->args[2];
+               arg.terse_dump = terse;
                tp->ops->walk(tp, &arg.w, true);
                cb->args[2] = arg.w.cookie;
                cb->args[1] = arg.w.count + 1;
        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 */
 static int tc_dump_tfilter(struct sk_buff *skb, struct netlink_callback *cb)
 {
        struct Qdisc *q = NULL;
        struct tcf_block *block;
        struct tcmsg *tcm = nlmsg_data(cb->nlh);
+       bool terse_dump = false;
        long index_start;
        long index;
        u32 parent;
                return skb->len;
 
        err = nlmsg_parse_deprecated(cb->nlh, sizeof(*tcm), tca, TCA_MAX,
-                                    NULL, cb->extack);
+                                    tcf_tfilter_dump_policy, cb->extack);
        if (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) {
                block = tcf_block_refcnt_get(net, tcm->tcm_block_index);
                if (!block)
                    nla_get_u32(tca[TCA_CHAIN]) != chain->index)
                        continue;
                if (!tcf_chain_dump(chain, q, parent, skb, cb,
-                                   index_start, &index)) {
+                                   index_start, &index, terse_dump)) {
                        tcf_chain_put(chain);
                        err = -EMSGSIZE;
                        break;