]> granicus.if.org Git - strace/blobdiff - bpf.c
arm, sparc, sparc64: wire up io_pgetevents
[strace] / bpf.c
diff --git a/bpf.c b/bpf.c
index caf04daebc7506e94f75f0db0e591148077f8913..e5dc4eeb1bd037aeebd96b62e8cb3fc5c78d656a 100644 (file)
--- a/bpf.c
+++ b/bpf.c
@@ -161,10 +161,10 @@ print_ebpf_insn(struct tcb * const tcp, void * const elem_buf,
        return true;
 }
 
-void
-print_ebpf_prog(struct tcb *const tcp, const kernel_ulong_t addr,
-               const uint32_t len)
+static void
+print_ebpf_prog(struct tcb *const tcp, const uint64_t addr, const uint32_t len)
 {
+       print_big_u64_addr(addr);
        if (abbrev(tcp)) {
                printaddr(addr);
        } else {
@@ -172,7 +172,7 @@ print_ebpf_prog(struct tcb *const tcp, const kernel_ulong_t addr,
                struct ebpf_insn insn;
 
                print_array(tcp, addr, len, &insn, sizeof(insn),
-                           umoven_or_printaddr, print_ebpf_insn, &eid);
+                           tfetch_mem, print_ebpf_insn, &eid);
        }
 }
 
@@ -274,7 +274,6 @@ BEGIN_BPF_CMD_DECODER(BPF_PROG_LOAD)
                               "BPF_PROG_TYPE_???");
        PRINT_FIELD_U(", ", attr, insn_cnt);
        tprints(", insns=");
-       print_big_u64_addr(attr.insns);
        print_ebpf_prog(tcp, attr.insns, attr.insn_cnt);
 
        tprintf(", license=");
@@ -286,7 +285,9 @@ BEGIN_BPF_CMD_DECODER(BPF_PROG_LOAD)
                break;
        PRINT_FIELD_U(", ", attr, log_level);
        PRINT_FIELD_U(", ", attr, log_size);
-       PRINT_FIELD_ADDR64(", ", attr, log_buf);
+       tprintf(", log_buf=");
+       print_big_u64_addr(attr.log_buf);
+       printstr_ex(tcp, attr.log_buf, attr.log_size, QUOTE_0_TERMINATED);
 
        /* kern_version field was added in Linux commit v4.1-rc1~84^2~50.  */
        if (len <= offsetof(struct BPF_PROG_LOAD_struct, kern_version))
@@ -421,21 +422,34 @@ BEGIN_BPF_CMD_DECODER(BPF_MAP_GET_FD_BY_ID)
 }
 END_BPF_CMD_DECODER(RVAL_DECODED)
 
-typedef bool (*print_bpf_obj_info_fn)(struct tcb *,
+struct obj_get_info_saved;
+typedef void (*print_bpf_obj_info_fn)(struct tcb *,
                                      uint32_t bpf_fd,
                                      const char *info_buf,
-                                     uint32_t size);
+                                     uint32_t size,
+                                     struct obj_get_info_saved *saved);
 
-static bool
+struct obj_get_info_saved {
+       print_bpf_obj_info_fn print_fn;
+
+       uint32_t info_len;
+
+       uint32_t jited_prog_len;
+       uint32_t xlated_prog_len;
+       uint32_t nr_map_ids;
+};
+
+static void
 print_bpf_map_info(struct tcb * const tcp, uint32_t bpf_fd,
-                  const char *info_buf, uint32_t size)
+                  const char *info_buf, uint32_t size,
+                  struct obj_get_info_saved *saved)
 {
+       if (entering(tcp))
+               return;
+
        struct bpf_map_info_struct info = { 0 };
        const unsigned int len = MIN(size, bpf_map_info_struct_size);
 
-       if (entering(tcp))
-               return false;
-
        memcpy(&info, info_buf, len);
 
        PRINT_FIELD_XVAL("{", info, type, bpf_map_types, "BPF_MAP_TYPE_???");
@@ -444,20 +458,34 @@ print_bpf_map_info(struct tcb * const tcp, uint32_t bpf_fd,
        PRINT_FIELD_U(", ", info, value_size);
        PRINT_FIELD_U(", ", info, max_entries);
        PRINT_FIELD_FLAGS(", ", info, map_flags, bpf_map_flags, "BPF_F_???");
+
+       /*
+        * "name" field was introduced by Linux commit v4.15-rc1~84^2~605^2~3.
+        */
+       if (len <= offsetof(struct bpf_map_info_struct, name))
+               goto print_bpf_map_info_end;
        PRINT_FIELD_CSTRING(", ", info, name);
+
+       /*
+        * ifindex, netns_dev, and netns_ino fields were introduced
+        * by Linux commit v4.16-rc1~123^2~109^2~5^2~4.
+        */
+       if (len <= offsetof(struct bpf_map_info_struct, ifindex))
+               goto print_bpf_map_info_end;
        PRINT_FIELD_IFINDEX(", ", info, ifindex);
        PRINT_FIELD_DEV(", ", info, netns_dev);
        PRINT_FIELD_U(", ", info, netns_ino);
 
        decode_attr_extra_data(tcp, info_buf, size, bpf_map_info_struct_size);
-       tprints("}");
 
-       return true;
+print_bpf_map_info_end:
+       tprints("}");
 }
 
-static bool
+static void
 print_bpf_prog_info(struct tcb * const tcp, uint32_t bpf_fd,
-                   const char *info_buf, uint32_t size)
+                   const char *info_buf, uint32_t size,
+                   struct obj_get_info_saved *saved)
 {
        struct bpf_prog_info_struct info = { 0 };
        const unsigned int len = MIN(size, bpf_prog_info_struct_size);
@@ -465,25 +493,14 @@ print_bpf_prog_info(struct tcb * const tcp, uint32_t bpf_fd,
 
        memcpy(&info, info_buf, len);
 
-       struct {
-               print_bpf_obj_info_fn print_fn;
-               uint32_t jited_prog_len;
-               uint32_t xlated_prog_len;
-               uint32_t nr_map_ids;
-       } *saved;
-
        if (entering(tcp)) {
-               saved = xcalloc(1, sizeof(*saved));
-               saved->print_fn = print_bpf_prog_info;
                saved->jited_prog_len = info.jited_prog_len;
                saved->xlated_prog_len = info.xlated_prog_len;
                saved->nr_map_ids = info.nr_map_ids;
 
-               return false;
+               return;
        }
 
-       saved = get_tcb_priv_data(tcp);
-
        PRINT_FIELD_XVAL("{", info, type, bpf_prog_types, "BPF_PROG_TYPE_???");
        PRINT_FIELD_U(", ", info, id);
        PRINT_FIELD_HEX_ARRAY(", ", info, tag);
@@ -504,9 +521,15 @@ print_bpf_prog_info(struct tcb * const tcp, uint32_t bpf_fd,
        tprintf("%" PRIu32, info.xlated_prog_len);
 
        tprints(", xlated_prog_insns=");
-       print_big_u64_addr(info.xlated_prog_insns);
-       print_ebpf_prog(tcp, info.xlated_prog_insns, info.xlated_prog_len / 8);
+       print_ebpf_prog(tcp, info.xlated_prog_insns,
+                       MIN(saved->xlated_prog_len, info.xlated_prog_len) / 8);
 
+       /*
+        * load_time, created_by_uid, nr_map_ids, map_ids, and name fields
+        * were introduced by Linux commit v4.15-rc1~84^2~605^2~4.
+        */
+       if (len <= offsetof(struct bpf_prog_info_struct, load_time))
+               goto print_bpf_prog_info_end;
        PRINT_FIELD_U(", ", info, load_time);
        PRINT_FIELD_UID(", ", info, created_by_uid);
 
@@ -517,18 +540,26 @@ print_bpf_prog_info(struct tcb * const tcp, uint32_t bpf_fd,
 
        tprints(", map_ids=");
        print_big_u64_addr(info.map_ids);
-       print_array(tcp, info.map_ids, info.nr_map_ids,
+       print_array(tcp, info.map_ids, MIN(saved->nr_map_ids, info.nr_map_ids),
                    &map_id_buf, sizeof(map_id_buf),
-                   umoven_or_printaddr, print_uint64_array_member, 0);
+                   tfetch_mem, print_uint32_array_member, 0);
+
+       PRINT_FIELD_CSTRING(", ", info, name);
 
+       /*
+        * ifindex, netns_dev, and netns_ino fields were introduced
+        * by Linux commit v4.16-rc1~123^2~227^2~5^2~2.
+        */
+       if (len <= offsetof(struct bpf_prog_info_struct, ifindex))
+               goto print_bpf_prog_info_end;
        PRINT_FIELD_IFINDEX(", ", info, ifindex);
        PRINT_FIELD_DEV(", ", info, netns_dev);
        PRINT_FIELD_U(", ", info, netns_ino);
 
        decode_attr_extra_data(tcp, info_buf, size, bpf_prog_info_struct_size);
-       tprints("}");
 
-       return true;
+print_bpf_prog_info_end:
+       tprints("}");
 }
 
 static const char *
@@ -548,19 +579,21 @@ fetch_bpf_obj_info(struct tcb * const tcp, uint64_t info, uint32_t size)
        return NULL;
 }
 
-static bool
-print_bpf_obj_info_addr(uint64_t addr)
+static void
+print_bpf_obj_info_addr(struct tcb * const tcp, uint64_t addr)
 {
-       printaddr64(addr);
-       return true;
+       if (exiting(tcp))
+               printaddr64(addr);
 }
 
-static bool
+static void
 print_bpf_obj_info(struct tcb * const tcp, uint32_t bpf_fd, uint64_t info,
-                  uint32_t size)
+                  uint32_t size, struct obj_get_info_saved *saved)
 {
-       if (abbrev(tcp))
-               return print_bpf_obj_info_addr(info);
+       if (abbrev(tcp)) {
+               print_bpf_obj_info_addr(tcp, info);
+               return;
+       }
 
        static struct {
                const char *id;
@@ -570,45 +603,56 @@ print_bpf_obj_info(struct tcb * const tcp, uint32_t bpf_fd, uint64_t info,
                { "anon_inode:bpf-prog", print_bpf_prog_info }
        };
 
-       print_bpf_obj_info_fn print_fn = NULL;
-       unsigned long i;
-
-       if (exiting(tcp)) {
-               i = get_tcb_priv_ulong(tcp);
-               print_fn = i < ARRAY_SIZE(obj_printers)
-                          ? obj_printers[i].print_fn
-                          : * (print_bpf_obj_info_fn *) get_tcb_priv_data(tcp);
-       } else {
+       if (entering(tcp)) {
                char path[PATH_MAX + 1];
 
                if (getfdpath(tcp, bpf_fd, path, sizeof(path)) > 0) {
-                       for (i = 0; i < ARRAY_SIZE(obj_printers); ++i) {
+                       for (size_t i = 0; i < ARRAY_SIZE(obj_printers); ++i) {
                                if (!strcmp(path, obj_printers[i].id)) {
-                                       print_fn = obj_printers[i].print_fn;
+                                       saved->print_fn =
+                                               obj_printers[i].print_fn;
                                        break;
                                }
                        }
                }
        }
 
-       if (!print_fn)
-               return print_bpf_obj_info_addr(info);
+       if (!saved || !saved->print_fn) {
+               print_bpf_obj_info_addr(tcp, info);
+               return;
+       }
 
        const char *info_buf = fetch_bpf_obj_info(tcp, info, size);
 
-       return info_buf ? print_fn(tcp, bpf_fd, info_buf, size)
-                       : print_bpf_obj_info_addr(info);
+       if (info_buf)
+               saved->print_fn(tcp, bpf_fd, info_buf, size, saved);
+       else
+               print_bpf_obj_info_addr(tcp, info);
 }
 
 BEGIN_BPF_CMD_DECODER(BPF_OBJ_GET_INFO_BY_FD)
 {
+       struct obj_get_info_saved *saved;
+
        if (entering(tcp)) {
+               saved = xcalloc(1, sizeof(*saved));
+               saved->info_len = attr.info_len;
+               set_tcb_priv_data(tcp, saved, free);
+
                PRINT_FIELD_FD("{info={", attr, bpf_fd, tcp);
                PRINT_FIELD_U(", ", attr, info_len);
+       } else {
+               saved = get_tcb_priv_data(tcp);
+
+               if (saved && (saved->info_len != attr.info_len))
+                       tprintf(" => %u", attr.info_len);
+
                tprintf(", info=");
        }
 
-       if (!print_bpf_obj_info(tcp, attr.bpf_fd, attr.info, attr.info_len))
+       print_bpf_obj_info(tcp, attr.bpf_fd, attr.info, attr.info_len, saved);
+
+       if (entering(tcp))
                return 0;
 
        tprints("}");
@@ -637,7 +681,7 @@ BEGIN_BPF_CMD_DECODER(BPF_PROG_QUERY)
 
        print_big_u64_addr(attr.prog_ids);
        print_array(tcp, attr.prog_ids, attr.prog_cnt, &prog_id_buf,
-                   sizeof(prog_id_buf), umoven_or_printaddr,
+                   sizeof(prog_id_buf), tfetch_mem,
                    print_uint32_array_member, 0);
 
        tprints(", prog_cnt=");