=============
 
 | **bpftool** **btf** { **show** | **list** } [**id** *BTF_ID*]
-| **bpftool** **btf dump** *BTF_SRC* [**format** *FORMAT*]
+| **bpftool** **btf dump** *BTF_SRC* [**format** *FORMAT*] [**root_id** *ROOT_ID*]
 | **bpftool** **btf help**
 |
 | *BTF_SRC* := { **id** *BTF_ID* | **prog** *PROG* | **map** *MAP* [{**key** | **value** | **kv** | **all**}] | **file** *FILE* }
     that hold open file descriptors (FDs) against BTF objects. On such kernels
     bpftool will automatically emit this information as well.
 
-bpftool btf dump *BTF_SRC* [format *FORMAT*]
+bpftool btf dump *BTF_SRC* [format *FORMAT*] [root_id *ROOT_ID*]
     Dump BTF entries from a given *BTF_SRC*.
 
     When **id** is specified, BTF object with that ID will be loaded and all
     formatting, the output is sorted by default. Use the **unsorted** option
     to avoid sorting the output.
 
+    **root_id** option can be used to filter a dump to a single type and all
+    its dependent types. It cannot be used with any other types of filtering
+    (such as the "key", "value", or "kv" arguments when dumping BTF for a map).
+    It can be passed multiple times to dump multiple types.
+
 bpftool btf help
     Print short help message.
 
 
 #define KFUNC_DECL_TAG         "bpf_kfunc"
 #define FASTCALL_DECL_TAG      "bpf_fastcall"
 
+#define MAX_ROOT_IDS           16
+
 static const char * const btf_kind_str[NR_BTF_KINDS] = {
        [BTF_KIND_UNKN]         = "UNKNOWN",
        [BTF_KIND_INT]          = "INT",
 {
        bool dump_c = false, sort_dump_c = true;
        struct btf *btf = NULL, *base = NULL;
-       __u32 root_type_ids[2];
+       __u32 root_type_ids[MAX_ROOT_IDS];
+       bool have_id_filtering;
        int root_type_cnt = 0;
        __u32 btf_id = -1;
        const char *src;
                goto done;
        }
 
+       have_id_filtering = !!root_type_cnt;
+
        while (argc) {
                if (is_prefix(*argv, "format")) {
                        NEXT_ARG();
                                goto done;
                        }
                        NEXT_ARG();
+               } else if (is_prefix(*argv, "root_id")) {
+                       __u32 root_id;
+                       char *end;
+
+                       if (have_id_filtering) {
+                               p_err("cannot use root_id with other type filtering");
+                               err = -EINVAL;
+                               goto done;
+                       } else if (root_type_cnt == MAX_ROOT_IDS) {
+                               p_err("only %d root_id are supported", MAX_ROOT_IDS);
+                               err = -E2BIG;
+                               goto done;
+                       }
+
+                       NEXT_ARG();
+                       root_id = strtoul(*argv, &end, 0);
+                       if (*end) {
+                               err = -1;
+                               p_err("can't parse %s as root ID", *argv);
+                               goto done;
+                       }
+                       for (i = 0; i < root_type_cnt; i++) {
+                               if (root_type_ids[i] == root_id) {
+                                       err = -EINVAL;
+                                       p_err("duplicate root_id %d supplied", root_id);
+                                       goto done;
+                               }
+                       }
+                       root_type_ids[root_type_cnt++] = root_id;
+                       NEXT_ARG();
                } else if (is_prefix(*argv, "unsorted")) {
                        sort_dump_c = false;
                        NEXT_ARG();
 
        fprintf(stderr,
                "Usage: %1$s %2$s { show | list } [id BTF_ID]\n"
-               "       %1$s %2$s dump BTF_SRC [format FORMAT]\n"
+               "       %1$s %2$s dump BTF_SRC [format FORMAT] [root_id ROOT_ID]\n"
                "       %1$s %2$s help\n"
                "\n"
                "       BTF_SRC := { id BTF_ID | prog PROG | map MAP [{key | value | kv | all}] | file FILE }\n"