diff options
Diffstat (limited to 'tools/bpf')
25 files changed, 1295 insertions, 692 deletions
| diff --git a/tools/bpf/bpftool/Documentation/bpftool-map.rst b/tools/bpf/bpftool/Documentation/bpftool-map.rst index 7bb787cfa971..64b001b4f777 100644 --- a/tools/bpf/bpftool/Documentation/bpftool-map.rst +++ b/tools/bpf/bpftool/Documentation/bpftool-map.rst @@ -42,7 +42,8 @@ MAP COMMANDS  |		| **percpu_array** | **stack_trace** | **cgroup_array** | **lru_hash**  |		| **lru_percpu_hash** | **lpm_trie** | **array_of_maps** | **hash_of_maps**  |		| **devmap** | **sockmap** | **cpumap** | **xskmap** | **sockhash** -|		| **cgroup_storage** | **reuseport_sockarray** | **percpu_cgroup_storage** } +|		| **cgroup_storage** | **reuseport_sockarray** | **percpu_cgroup_storage** +|		| **queue** | **stack** }  DESCRIPTION  =========== @@ -127,6 +128,10 @@ OPTIONS  	-f, --bpffs  		  Show file names of pinned maps. +	-n, --nomount +		  Do not automatically attempt to mount any virtual file system +		  (such as tracefs or BPF virtual file system) when necessary. +  EXAMPLES  ========  **# bpftool map show** @@ -169,6 +174,61 @@ The following three commands are equivalent:  | **# bpftool map pin id 10 /sys/fs/bpf/map**  | **# bpftool map del pinned /sys/fs/bpf/map key 13 00 07 00** +Note that map update can also be used in order to change the program references +hold by a program array map. This can be used, for example, to change the +programs used for tail-call jumps at runtime, without having to reload the +entry-point program. Below is an example for this use case: we load a program +defining a prog array map, and with a main function that contains a tail call +to other programs that can be used either to "process" packets or to "debug" +processing. Note that the prog array map MUST be pinned into the BPF virtual +file system for the map update to work successfully, as kernel flushes prog +array maps when they have no more references from user space (and the update +would be lost as soon as bpftool exits). + +| +| **# bpftool prog loadall tail_calls.o /sys/fs/bpf/foo type xdp** +| **# bpftool prog --bpffs** + +:: + +  545: xdp  name main_func  tag 674b4b5597193dc3  gpl +          loaded_at 2018-12-12T15:02:58+0000  uid 0 +          xlated 240B  jited 257B  memlock 4096B  map_ids 294 +          pinned /sys/fs/bpf/foo/xdp +  546: xdp  name bpf_func_process  tag e369a529024751fc  gpl +          loaded_at 2018-12-12T15:02:58+0000  uid 0 +          xlated 200B  jited 164B  memlock 4096B +          pinned /sys/fs/bpf/foo/process +  547: xdp  name bpf_func_debug  tag 0b597868bc7f0976  gpl +          loaded_at 2018-12-12T15:02:58+0000  uid 0 +          xlated 200B  jited 164B  memlock 4096B +          pinned /sys/fs/bpf/foo/debug + +**# bpftool map** + +:: + +  294: prog_array  name jmp_table  flags 0x0 +          key 4B  value 4B  max_entries 1  memlock 4096B +          owner_prog_type xdp  owner jited + +| +| **# bpftool map pin id 294 /sys/fs/bpf/bar** +| **# bpftool map dump pinned /sys/fs/bpf/bar** + +:: + +  Found 0 elements + +| +| **# bpftool map update pinned /sys/fs/bpf/bar key 0 0 0 0 value pinned /sys/fs/bpf/foo/debug** +| **# bpftool map dump pinned /sys/fs/bpf/bar** + +:: + +  key: 00 00 00 00  value: 22 02 00 00 +  Found 1 element +  SEE ALSO  ========  	**bpf**\ (2), diff --git a/tools/bpf/bpftool/Documentation/bpftool-prog.rst b/tools/bpf/bpftool/Documentation/bpftool-prog.rst index ecf618807125..58c8369b77dd 100644 --- a/tools/bpf/bpftool/Documentation/bpftool-prog.rst +++ b/tools/bpf/bpftool/Documentation/bpftool-prog.rst @@ -15,18 +15,20 @@ SYNOPSIS  	*OPTIONS* := { { **-j** | **--json** } [{ **-p** | **--pretty** }] | { **-f** | **--bpffs** } }  	*COMMANDS* := -	{ **show** | **list** | **dump xlated** | **dump jited** | **pin** | **load** | **help** } +	{ **show** | **list** | **dump xlated** | **dump jited** | **pin** | **load** +	| **loadall** | **help** }  MAP COMMANDS  =============  |	**bpftool** **prog { show | list }** [*PROG*] -|	**bpftool** **prog dump xlated** *PROG* [{**file** *FILE* | **opcodes** | **visual**}] -|	**bpftool** **prog dump jited**  *PROG* [{**file** *FILE* | **opcodes**}] +|	**bpftool** **prog dump xlated** *PROG* [{**file** *FILE* | **opcodes** | **visual** | **linum**}] +|	**bpftool** **prog dump jited**  *PROG* [{**file** *FILE* | **opcodes** | **linum**}]  |	**bpftool** **prog pin** *PROG* *FILE* -|	**bpftool** **prog load** *OBJ* *FILE* [**type** *TYPE*] [**map** {**idx** *IDX* | **name** *NAME*} *MAP*] [**dev** *NAME*] -|       **bpftool** **prog attach** *PROG* *ATTACH_TYPE* *MAP* -|       **bpftool** **prog detach** *PROG* *ATTACH_TYPE* *MAP* +|	**bpftool** **prog { load | loadall }** *OBJ* *PATH* [**type** *TYPE*] [**map** {**idx** *IDX* | **name** *NAME*} *MAP*] [**dev** *NAME*] +|	**bpftool** **prog attach** *PROG* *ATTACH_TYPE* [*MAP*] +|	**bpftool** **prog detach** *PROG* *ATTACH_TYPE* [*MAP*] +|	**bpftool** **prog tracelog**  |	**bpftool** **prog help**  |  |	*MAP* := { **id** *MAP_ID* | **pinned** *FILE* } @@ -39,7 +41,9 @@ MAP COMMANDS  |		**cgroup/bind4** | **cgroup/bind6** | **cgroup/post_bind4** | **cgroup/post_bind6** |  |		**cgroup/connect4** | **cgroup/connect6** | **cgroup/sendmsg4** | **cgroup/sendmsg6**  |	} -|       *ATTACH_TYPE* := { **msg_verdict** | **skb_verdict** | **skb_parse** } +|       *ATTACH_TYPE* := { +|		**msg_verdict** | **skb_verdict** | **skb_parse** | **flow_dissector** +|	}  DESCRIPTION @@ -52,7 +56,7 @@ DESCRIPTION  		  Output will start with program ID followed by program type and  		  zero or more named attributes (depending on kernel version). -	**bpftool prog dump xlated** *PROG* [{ **file** *FILE* | **opcodes** | **visual** }] +	**bpftool prog dump xlated** *PROG* [{ **file** *FILE* | **opcodes** | **visual** | **linum** }]  		  Dump eBPF instructions of the program from the kernel. By  		  default, eBPF will be disassembled and printed to standard  		  output in human-readable format. In this case, **opcodes** @@ -65,13 +69,23 @@ DESCRIPTION  		  built instead, and eBPF instructions will be presented with  		  CFG in DOT format, on standard output. -	**bpftool prog dump jited**  *PROG* [{ **file** *FILE* | **opcodes** }] +		  If the prog has line_info available, the source line will +		  be displayed by default.  If **linum** is specified, +		  the filename, line number and line column will also be +		  displayed on top of the source line. + +	**bpftool prog dump jited**  *PROG* [{ **file** *FILE* | **opcodes** | **linum** }]  		  Dump jited image (host machine code) of the program.  		  If *FILE* is specified image will be written to a file,  		  otherwise it will be disassembled and printed to stdout.  		  **opcodes** controls if raw opcodes will be printed. +		  If the prog has line_info available, the source line will +		  be displayed by default.  If **linum** is specified, +		  the filename, line number and line column will also be +		  displayed on top of the source line. +  	**bpftool prog pin** *PROG* *FILE*  		  Pin program *PROG* as *FILE*. @@ -79,8 +93,11 @@ DESCRIPTION  		  contain a dot character ('.'), which is reserved for future  		  extensions of *bpffs*. -	**bpftool prog load** *OBJ* *FILE* [**type** *TYPE*] [**map** {**idx** *IDX* | **name** *NAME*} *MAP*] [**dev** *NAME*] -		  Load bpf program from binary *OBJ* and pin as *FILE*. +	**bpftool prog { load | loadall }** *OBJ* *PATH* [**type** *TYPE*] [**map** {**idx** *IDX* | **name** *NAME*} *MAP*] [**dev** *NAME*] [**pinmaps** *MAP_DIR*] +		  Load bpf program(s) from binary *OBJ* and pin as *PATH*. +		  **bpftool prog load** pins only the first program from the +		  *OBJ* as *PATH*. **bpftool prog loadall** pins all programs +		  from the *OBJ* under *PATH* directory.  		  **type** is optional, if not specified program type will be  		  inferred from section names.  		  By default bpftool will create new maps as declared in the ELF @@ -92,18 +109,32 @@ DESCRIPTION  		  use, referring to it by **id** or through a **pinned** file.  		  If **dev** *NAME* is specified program will be loaded onto  		  given networking device (offload). +		  Optional **pinmaps** argument can be provided to pin all +		  maps under *MAP_DIR* directory. -		  Note: *FILE* must be located in *bpffs* mount. It must not +		  Note: *PATH* must be located in *bpffs* mount. It must not  		  contain a dot character ('.'), which is reserved for future  		  extensions of *bpffs*. -        **bpftool prog attach** *PROG* *ATTACH_TYPE* *MAP* -                  Attach bpf program *PROG* (with type specified by *ATTACH_TYPE*) -                  to the map *MAP*. - -        **bpftool prog detach** *PROG* *ATTACH_TYPE* *MAP* -                  Detach bpf program *PROG* (with type specified by *ATTACH_TYPE*) -                  from the map *MAP*. +	**bpftool prog attach** *PROG* *ATTACH_TYPE* [*MAP*] +		  Attach bpf program *PROG* (with type specified by +		  *ATTACH_TYPE*). Most *ATTACH_TYPEs* require a *MAP* +		  parameter, with the exception of *flow_dissector* which is +		  attached to current networking name space. + +	**bpftool prog detach** *PROG* *ATTACH_TYPE* [*MAP*] +		  Detach bpf program *PROG* (with type specified by +		  *ATTACH_TYPE*). Most *ATTACH_TYPEs* require a *MAP* +		  parameter, with the exception of *flow_dissector* which is +		  detached from the current networking name space. + +	**bpftool prog tracelog** +		  Dump the trace pipe of the system to the console (stdout). +		  Hit <Ctrl+C> to stop printing. BPF programs can write to this +		  trace pipe at runtime with the **bpf_trace_printk()** helper. +		  This should be used only for debugging purposes. For +		  streaming data from BPF programs to user space, one can use +		  perf events (see also **bpftool-map**\ (8)).  	**bpftool prog help**  		  Print short help message. @@ -127,83 +158,98 @@ OPTIONS  		  When showing BPF programs, show file names of pinned  		  programs. +	-m, --mapcompat +		  Allow loading maps with unknown map definitions. + +	-n, --nomount +		  Do not automatically attempt to mount any virtual file system +		  (such as tracefs or BPF virtual file system) when necessary. +  EXAMPLES  ========  **# bpftool prog show** +  :: -  10: xdp  name some_prog  tag 005a3d2123620c8b  gpl -	loaded_at Sep 29/20:11  uid 0 -	xlated 528B  jited 370B  memlock 4096B  map_ids 10 +    10: xdp  name some_prog  tag 005a3d2123620c8b  gpl +            loaded_at 2017-09-29T20:11:00+0000  uid 0 +            xlated 528B  jited 370B  memlock 4096B  map_ids 10  **# bpftool --json --pretty prog show**  :: -    { -        "programs": [{ -                "id": 10, -                "type": "xdp", -                "tag": "005a3d2123620c8b", -                "gpl_compatible": true, -                "loaded_at": "Sep 29/20:11", -                "uid": 0, -                "bytes_xlated": 528, -                "jited": true, -                "bytes_jited": 370, -                "bytes_memlock": 4096, -                "map_ids": [10 -                ] -            } -        ] -    } +    [{ +            "id": 10, +            "type": "xdp", +            "tag": "005a3d2123620c8b", +            "gpl_compatible": true, +            "loaded_at": 1506715860, +            "uid": 0, +            "bytes_xlated": 528, +            "jited": true, +            "bytes_jited": 370, +            "bytes_memlock": 4096, +            "map_ids": [10 +            ] +        } +    ]  |  | **# bpftool prog dump xlated id 10 file /tmp/t**  | **# ls -l /tmp/t** -|   -rw------- 1 root root 560 Jul 22 01:42 /tmp/t -**# bpftool prog dum jited tag 005a3d2123620c8b** +:: + +    -rw------- 1 root root 560 Jul 22 01:42 /tmp/t + +**# bpftool prog dump jited tag 005a3d2123620c8b**  :: -    push   %rbp -    mov    %rsp,%rbp -    sub    $0x228,%rsp -    sub    $0x28,%rbp -    mov    %rbx,0x0(%rbp) +    0:   push   %rbp +    1:   mov    %rsp,%rbp +    2:   sub    $0x228,%rsp +    3:   sub    $0x28,%rbp +    4:   mov    %rbx,0x0(%rbp)  |  | **# mount -t bpf none /sys/fs/bpf/**  | **# bpftool prog pin id 10 /sys/fs/bpf/prog**  | **# bpftool prog load ./my_prog.o /sys/fs/bpf/prog2**  | **# ls -l /sys/fs/bpf/** -|   -rw------- 1 root root 0 Jul 22 01:43 prog -|   -rw------- 1 root root 0 Jul 22 01:44 prog2 -**# bpftool prog dum jited pinned /sys/fs/bpf/prog opcodes** +:: + +    -rw------- 1 root root 0 Jul 22 01:43 prog +    -rw------- 1 root root 0 Jul 22 01:44 prog2 + +**# bpftool prog dump jited pinned /sys/fs/bpf/prog opcodes**  :: -    push   %rbp -    55 -    mov    %rsp,%rbp -    48 89 e5 -    sub    $0x228,%rsp -    48 81 ec 28 02 00 00 -    sub    $0x28,%rbp -    48 83 ed 28 -    mov    %rbx,0x0(%rbp) -    48 89 5d 00 +   0:   push   %rbp +        55 +   1:   mov    %rsp,%rbp +        48 89 e5 +   4:   sub    $0x228,%rsp +        48 81 ec 28 02 00 00 +   b:   sub    $0x28,%rbp +        48 83 ed 28 +   f:   mov    %rbx,0x0(%rbp) +        48 89 5d 00  |  | **# bpftool prog load xdp1_kern.o /sys/fs/bpf/xdp1 type xdp map name rxcnt id 7**  | **# bpftool prog show pinned /sys/fs/bpf/xdp1** -|   9: xdp  name xdp_prog1  tag 539ec6ce11b52f98  gpl -|	loaded_at 2018-06-25T16:17:31-0700  uid 0 -|	xlated 488B  jited 336B  memlock 4096B  map_ids 7 -| **# rm /sys/fs/bpf/xdp1** -| + +:: + +    9: xdp  name xdp_prog1  tag 539ec6ce11b52f98  gpl +            loaded_at 2018-06-25T16:17:31-0700  uid 0 +            xlated 488B  jited 336B  memlock 4096B  map_ids 7 + +**# rm /sys/fs/bpf/xdp1**  SEE ALSO  ======== diff --git a/tools/bpf/bpftool/Documentation/bpftool.rst b/tools/bpf/bpftool/Documentation/bpftool.rst index 129b7a9c0f9b..e1677e81ed59 100644 --- a/tools/bpf/bpftool/Documentation/bpftool.rst +++ b/tools/bpf/bpftool/Documentation/bpftool.rst @@ -60,6 +60,10 @@ OPTIONS  	-m, --mapcompat  		  Allow loading maps with unknown map definitions. +	-n, --nomount +		  Do not automatically attempt to mount any virtual file system +		  (such as tracefs or BPF virtual file system) when necessary. +  SEE ALSO  ======== diff --git a/tools/bpf/bpftool/Makefile b/tools/bpf/bpftool/Makefile index dac7eff4c7e5..492f0f24e2d3 100644 --- a/tools/bpf/bpftool/Makefile +++ b/tools/bpf/bpftool/Makefile @@ -35,8 +35,6 @@ $(LIBBPF)-clean:  prefix ?= /usr/local  bash_compdir ?= /usr/share/bash-completion/completions -CC = gcc -  CFLAGS += -O2  CFLAGS += -W -Wall -Wextra -Wno-unused-parameter -Wshadow -Wno-missing-field-initializers  CFLAGS += -DPACKAGE='"bpftool"' -D__EXPORTED_HEADERS__ \ @@ -53,7 +51,7 @@ ifneq ($(EXTRA_LDFLAGS),)  LDFLAGS += $(EXTRA_LDFLAGS)  endif -LIBS = -lelf -lbfd -lopcodes $(LIBBPF) +LIBS = -lelf $(LIBBPF)  INSTALL ?= install  RM ?= rm -f @@ -90,7 +88,16 @@ include $(wildcard $(OUTPUT)*.d)  all: $(OUTPUT)bpftool -SRCS = $(wildcard *.c) +BFD_SRCS = jit_disasm.c + +SRCS = $(filter-out $(BFD_SRCS),$(wildcard *.c)) + +ifeq ($(feature-libbfd),1) +CFLAGS += -DHAVE_LIBBFD_SUPPORT +SRCS += $(BFD_SRCS) +LIBS += -lbfd -lopcodes +endif +  OBJS = $(patsubst %.c,$(OUTPUT)%.o,$(SRCS)) $(OUTPUT)disasm.o  $(OUTPUT)disasm.o: $(srctree)/kernel/bpf/disasm.c diff --git a/tools/bpf/bpftool/bash-completion/bpftool b/tools/bpf/bpftool/bash-completion/bpftool index 3f78e6404589..e4e4fab1b8c7 100644 --- a/tools/bpf/bpftool/bash-completion/bpftool +++ b/tools/bpf/bpftool/bash-completion/bpftool @@ -1,37 +1,8 @@  # bpftool(8) bash completion                               -*- shell-script -*-  # +# SPDX-License-Identifier: (GPL-2.0-only OR BSD-2-Clause)  # Copyright (C) 2017-2018 Netronome Systems, Inc.  # -# This software is dual licensed under the GNU General License -# Version 2, June 1991 as shown in the file COPYING in the top-level -# directory of this source tree or the BSD 2-Clause License provided -# below.  You have the option to license this software under the -# complete terms of either license. -# -# The BSD 2-Clause License: -# -#     Redistribution and use in source and binary forms, with or -#     without modification, are permitted provided that the following -#     conditions are met: -# -#      1. Redistributions of source code must retain the above -#         copyright notice, this list of conditions and the following -#         disclaimer. -# -#      2. Redistributions in binary form must reproduce the above -#         copyright notice, this list of conditions and the following -#         disclaimer in the documentation and/or other materials -#         provided with the distribution. -# -# THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, -# EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF -# MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND -# NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS -# BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN -# ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN -# CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE -# SOFTWARE. -#  # Author: Quentin Monnet <quentin.monnet@netronome.com>  # Takes a list of words in argument; each one of them is added to COMPREPLY if @@ -191,7 +162,7 @@ _bpftool()      # Deal with simplest keywords      case $prev in -        help|hex|opcodes|visual) +        help|hex|opcodes|visual|linum)              return 0              ;;          tag) @@ -243,16 +214,20 @@ _bpftool()      # Completion depends on object and command in use      case $object in          prog) -            if [[ $command != "load" ]]; then -                case $prev in -                    id) -                        _bpftool_get_prog_ids -                        return 0 -                        ;; -                esac -            fi +            # Complete id, only for subcommands that use prog (but no map) ids +            case $command in +                show|list|dump|pin) +                    case $prev in +                        id) +                            _bpftool_get_prog_ids +                            return 0 +                            ;; +                    esac +                    ;; +            esac              local PROG_TYPE='id pinned tag' +            local MAP_TYPE='id pinned'              case $command in                  show|list)                      [[ $prev != "$command" ]] && return 0 @@ -274,10 +249,10 @@ _bpftool()                      *)                          _bpftool_once_attr 'file'                          if _bpftool_search_list 'xlated'; then -                            COMPREPLY+=( $( compgen -W 'opcodes visual' -- \ +                            COMPREPLY+=( $( compgen -W 'opcodes visual linum' -- \                                  "$cur" ) )                          else -                            COMPREPLY+=( $( compgen -W 'opcodes' -- \ +                            COMPREPLY+=( $( compgen -W 'opcodes linum' -- \                                  "$cur" ) )                          fi                          return 0 @@ -293,23 +268,45 @@ _bpftool()                      return 0                      ;;                  attach|detach) -                    if [[ ${#words[@]} == 7 ]]; then -                        COMPREPLY=( $( compgen -W "id pinned" -- "$cur" ) ) -                        return 0 -                    fi - -                    if [[ ${#words[@]} == 6 ]]; then -                        COMPREPLY=( $( compgen -W "msg_verdict skb_verdict skb_parse" -- "$cur" ) ) -                        return 0 -                    fi - -                    if [[ $prev == "$command" ]]; then -                        COMPREPLY=( $( compgen -W "id pinned" -- "$cur" ) ) -                        return 0 -                    fi -                    return 0 +                    case $cword in +                        3) +                            COMPREPLY=( $( compgen -W "$PROG_TYPE" -- "$cur" ) ) +                            return 0 +                            ;; +                        4) +                            case $prev in +                                id) +                                    _bpftool_get_prog_ids +                                    ;; +                                pinned) +                                    _filedir +                                    ;; +                            esac +                            return 0 +                            ;; +                        5) +                            COMPREPLY=( $( compgen -W 'msg_verdict skb_verdict \ +                                skb_parse flow_dissector' -- "$cur" ) ) +                            return 0 +                            ;; +                        6) +                            COMPREPLY=( $( compgen -W "$MAP_TYPE" -- "$cur" ) ) +                            return 0 +                            ;; +                        7) +                            case $prev in +                                id) +                                    _bpftool_get_map_ids +                                    ;; +                                pinned) +                                    _filedir +                                    ;; +                            esac +                            return 0 +                            ;; +                    esac                      ;; -                load) +                load|loadall)                      local obj                      if [[ ${#words[@]} -lt 6 ]]; then @@ -338,7 +335,16 @@ _bpftool()                      case $prev in                          type) -                            COMPREPLY=( $( compgen -W "socket kprobe kretprobe classifier action tracepoint raw_tracepoint xdp perf_event cgroup/skb cgroup/sock cgroup/dev lwt_in lwt_out lwt_xmit lwt_seg6local sockops sk_skb sk_msg lirc_mode2 cgroup/bind4 cgroup/bind6 cgroup/connect4 cgroup/connect6 cgroup/sendmsg4 cgroup/sendmsg6 cgroup/post_bind4 cgroup/post_bind6" -- \ +                            COMPREPLY=( $( compgen -W "socket kprobe \ +                                kretprobe classifier flow_dissector \ +                                action tracepoint raw_tracepoint \ +                                xdp perf_event cgroup/skb cgroup/sock \ +                                cgroup/dev lwt_in lwt_out lwt_xmit \ +                                lwt_seg6local sockops sk_skb sk_msg \ +                                lirc_mode2 cgroup/bind4 cgroup/bind6 \ +                                cgroup/connect4 cgroup/connect6 \ +                                cgroup/sendmsg4 cgroup/sendmsg6 \ +                                cgroup/post_bind4 cgroup/post_bind6" -- \                                                     "$cur" ) )                              return 0                              ;; @@ -346,7 +352,7 @@ _bpftool()                              _bpftool_get_map_ids                              return 0                              ;; -                        pinned) +                        pinned|pinmaps)                              _filedir                              return 0                              ;; @@ -358,14 +364,18 @@ _bpftool()                              COMPREPLY=( $( compgen -W "map" -- "$cur" ) )                              _bpftool_once_attr 'type'                              _bpftool_once_attr 'dev' +                            _bpftool_once_attr 'pinmaps'                              return 0                              ;;                      esac                      ;; +                tracelog) +                    return 0 +                    ;;                  *)                      [[ $prev == $object ]] && \                          COMPREPLY=( $( compgen -W 'dump help pin attach detach load \ -                            show list' -- "$cur" ) ) +                            show list tracelog' -- "$cur" ) )                      ;;              esac              ;; @@ -400,7 +410,7 @@ _bpftool()                                  lru_percpu_hash lpm_trie array_of_maps \                                  hash_of_maps devmap sockmap cpumap xskmap \                                  sockhash cgroup_storage reuseport_sockarray \ -                                percpu_cgroup_storage' -- \ +                                percpu_cgroup_storage queue stack' -- \                                                     "$cur" ) )                              return 0                              ;; diff --git a/tools/bpf/bpftool/btf_dumper.c b/tools/bpf/bpftool/btf_dumper.c index e4e6e2b3fd84..3f0629edbca5 100644 --- a/tools/bpf/bpftool/btf_dumper.c +++ b/tools/bpf/bpftool/btf_dumper.c @@ -1,4 +1,4 @@ -// SPDX-License-Identifier: GPL-2.0 +// SPDX-License-Identifier: (GPL-2.0-only OR BSD-2-Clause)  /* Copyright (c) 2018 Facebook */  #include <ctype.h> @@ -73,20 +73,17 @@ static int btf_dumper_array(const struct btf_dumper *d, __u32 type_id,  	return ret;  } -static void btf_dumper_int_bits(__u32 int_type, __u8 bit_offset, +static void btf_dumper_bitfield(__u32 nr_bits, __u8 bit_offset,  				const void *data, json_writer_t *jw,  				bool is_plain_text)  {  	int left_shift_bits, right_shift_bits; -	int nr_bits = BTF_INT_BITS(int_type); -	int total_bits_offset;  	int bytes_to_copy;  	int bits_to_copy;  	__u64 print_num; -	total_bits_offset = bit_offset + BTF_INT_OFFSET(int_type); -	data += BITS_ROUNDDOWN_BYTES(total_bits_offset); -	bit_offset = BITS_PER_BYTE_MASKED(total_bits_offset); +	data += BITS_ROUNDDOWN_BYTES(bit_offset); +	bit_offset = BITS_PER_BYTE_MASKED(bit_offset);  	bits_to_copy = bit_offset + nr_bits;  	bytes_to_copy = BITS_ROUNDUP_BYTES(bits_to_copy); @@ -109,6 +106,22 @@ static void btf_dumper_int_bits(__u32 int_type, __u8 bit_offset,  		jsonw_printf(jw, "%llu", print_num);  } + +static void btf_dumper_int_bits(__u32 int_type, __u8 bit_offset, +				const void *data, json_writer_t *jw, +				bool is_plain_text) +{ +	int nr_bits = BTF_INT_BITS(int_type); +	int total_bits_offset; + +	/* bits_offset is at most 7. +	 * BTF_INT_OFFSET() cannot exceed 64 bits. +	 */ +	total_bits_offset = bit_offset + BTF_INT_OFFSET(int_type); +	btf_dumper_bitfield(nr_bits, total_bits_offset, data, jw, +			    is_plain_text); +} +  static int btf_dumper_int(const struct btf_type *t, __u8 bit_offset,  			  const void *data, json_writer_t *jw,  			  bool is_plain_text) @@ -180,6 +193,7 @@ static int btf_dumper_struct(const struct btf_dumper *d, __u32 type_id,  	const struct btf_type *t;  	struct btf_member *m;  	const void *data_off; +	int kind_flag;  	int ret = 0;  	int i, vlen; @@ -187,18 +201,32 @@ static int btf_dumper_struct(const struct btf_dumper *d, __u32 type_id,  	if (!t)  		return -EINVAL; +	kind_flag = BTF_INFO_KFLAG(t->info);  	vlen = BTF_INFO_VLEN(t->info);  	jsonw_start_object(d->jw);  	m = (struct btf_member *)(t + 1);  	for (i = 0; i < vlen; i++) { -		data_off = data + BITS_ROUNDDOWN_BYTES(m[i].offset); +		__u32 bit_offset = m[i].offset; +		__u32 bitfield_size = 0; + +		if (kind_flag) { +			bitfield_size = BTF_MEMBER_BITFIELD_SIZE(bit_offset); +			bit_offset = BTF_MEMBER_BIT_OFFSET(bit_offset); +		} +  		jsonw_name(d->jw, btf__name_by_offset(d->btf, m[i].name_off)); -		ret = btf_dumper_do_type(d, m[i].type, -					 BITS_PER_BYTE_MASKED(m[i].offset), -					 data_off); -		if (ret) -			break; +		if (bitfield_size) { +			btf_dumper_bitfield(bitfield_size, bit_offset, +					    data, d->jw, d->is_plain_text); +		} else { +			data_off = data + BITS_ROUNDDOWN_BYTES(bit_offset); +			ret = btf_dumper_do_type(d, m[i].type, +						 BITS_PER_BYTE_MASKED(bit_offset), +						 data_off); +			if (ret) +				break; +		}  	}  	jsonw_end_object(d->jw); @@ -249,3 +277,206 @@ int btf_dumper_type(const struct btf_dumper *d, __u32 type_id,  {  	return btf_dumper_do_type(d, type_id, 0, data);  } + +#define BTF_PRINT_ARG(...)						\ +	do {								\ +		pos += snprintf(func_sig + pos, size - pos,		\ +				__VA_ARGS__);				\ +		if (pos >= size)					\ +			return -1;					\ +	} while (0) +#define BTF_PRINT_TYPE(type)					\ +	do {								\ +		pos = __btf_dumper_type_only(btf, type, func_sig,	\ +					     pos, size);		\ +		if (pos == -1)						\ +			return -1;					\ +	} while (0) + +static int btf_dump_func(const struct btf *btf, char *func_sig, +			 const struct btf_type *func_proto, +			 const struct btf_type *func, int pos, int size); + +static int __btf_dumper_type_only(const struct btf *btf, __u32 type_id, +				  char *func_sig, int pos, int size) +{ +	const struct btf_type *proto_type; +	const struct btf_array *array; +	const struct btf_type *t; + +	if (!type_id) { +		BTF_PRINT_ARG("void "); +		return pos; +	} + +	t = btf__type_by_id(btf, type_id); + +	switch (BTF_INFO_KIND(t->info)) { +	case BTF_KIND_INT: +	case BTF_KIND_TYPEDEF: +		BTF_PRINT_ARG("%s ", btf__name_by_offset(btf, t->name_off)); +		break; +	case BTF_KIND_STRUCT: +		BTF_PRINT_ARG("struct %s ", +			      btf__name_by_offset(btf, t->name_off)); +		break; +	case BTF_KIND_UNION: +		BTF_PRINT_ARG("union %s ", +			      btf__name_by_offset(btf, t->name_off)); +		break; +	case BTF_KIND_ENUM: +		BTF_PRINT_ARG("enum %s ", +			      btf__name_by_offset(btf, t->name_off)); +		break; +	case BTF_KIND_ARRAY: +		array = (struct btf_array *)(t + 1); +		BTF_PRINT_TYPE(array->type); +		BTF_PRINT_ARG("[%d]", array->nelems); +		break; +	case BTF_KIND_PTR: +		BTF_PRINT_TYPE(t->type); +		BTF_PRINT_ARG("* "); +		break; +	case BTF_KIND_FWD: +		BTF_PRINT_ARG("%s %s ", +			      BTF_INFO_KFLAG(t->info) ? "union" : "struct", +			      btf__name_by_offset(btf, t->name_off)); +		break; +	case BTF_KIND_VOLATILE: +		BTF_PRINT_ARG("volatile "); +		BTF_PRINT_TYPE(t->type); +		break; +	case BTF_KIND_CONST: +		BTF_PRINT_ARG("const "); +		BTF_PRINT_TYPE(t->type); +		break; +	case BTF_KIND_RESTRICT: +		BTF_PRINT_ARG("restrict "); +		BTF_PRINT_TYPE(t->type); +		break; +	case BTF_KIND_FUNC_PROTO: +		pos = btf_dump_func(btf, func_sig, t, NULL, pos, size); +		if (pos == -1) +			return -1; +		break; +	case BTF_KIND_FUNC: +		proto_type = btf__type_by_id(btf, t->type); +		pos = btf_dump_func(btf, func_sig, proto_type, t, pos, size); +		if (pos == -1) +			return -1; +		break; +	case BTF_KIND_UNKN: +	default: +		return -1; +	} + +	return pos; +} + +static int btf_dump_func(const struct btf *btf, char *func_sig, +			 const struct btf_type *func_proto, +			 const struct btf_type *func, int pos, int size) +{ +	int i, vlen; + +	BTF_PRINT_TYPE(func_proto->type); +	if (func) +		BTF_PRINT_ARG("%s(", btf__name_by_offset(btf, func->name_off)); +	else +		BTF_PRINT_ARG("("); +	vlen = BTF_INFO_VLEN(func_proto->info); +	for (i = 0; i < vlen; i++) { +		struct btf_param *arg = &((struct btf_param *)(func_proto + 1))[i]; + +		if (i) +			BTF_PRINT_ARG(", "); +		if (arg->type) { +			BTF_PRINT_TYPE(arg->type); +			BTF_PRINT_ARG("%s", +				      btf__name_by_offset(btf, arg->name_off)); +		} else { +			BTF_PRINT_ARG("..."); +		} +	} +	BTF_PRINT_ARG(")"); + +	return pos; +} + +void btf_dumper_type_only(const struct btf *btf, __u32 type_id, char *func_sig, +			  int size) +{ +	int err; + +	func_sig[0] = '\0'; +	if (!btf) +		return; + +	err = __btf_dumper_type_only(btf, type_id, func_sig, 0, size); +	if (err < 0) +		func_sig[0] = '\0'; +} + +static const char *ltrim(const char *s) +{ +	while (isspace(*s)) +		s++; + +	return s; +} + +void btf_dump_linfo_plain(const struct btf *btf, +			  const struct bpf_line_info *linfo, +			  const char *prefix, bool linum) +{ +	const char *line = btf__name_by_offset(btf, linfo->line_off); + +	if (!line) +		return; +	line = ltrim(line); + +	if (!prefix) +		prefix = ""; + +	if (linum) { +		const char *file = btf__name_by_offset(btf, linfo->file_name_off); + +		/* More forgiving on file because linum option is +		 * expected to provide more info than the already +		 * available src line. +		 */ +		if (!file) +			file = ""; + +		printf("%s%s [file:%s line_num:%u line_col:%u]\n", +		       prefix, line, file, +		       BPF_LINE_INFO_LINE_NUM(linfo->line_col), +		       BPF_LINE_INFO_LINE_COL(linfo->line_col)); +	} else { +		printf("%s%s\n", prefix, line); +	} +} + +void btf_dump_linfo_json(const struct btf *btf, +			 const struct bpf_line_info *linfo, bool linum) +{ +	const char *line = btf__name_by_offset(btf, linfo->line_off); + +	if (line) +		jsonw_string_field(json_wtr, "src", ltrim(line)); + +	if (linum) { +		const char *file = btf__name_by_offset(btf, linfo->file_name_off); + +		if (file) +			jsonw_string_field(json_wtr, "file", file); + +		if (BPF_LINE_INFO_LINE_NUM(linfo->line_col)) +			jsonw_int_field(json_wtr, "line_num", +					BPF_LINE_INFO_LINE_NUM(linfo->line_col)); + +		if (BPF_LINE_INFO_LINE_COL(linfo->line_col)) +			jsonw_int_field(json_wtr, "line_col", +					BPF_LINE_INFO_LINE_COL(linfo->line_col)); +	} +} diff --git a/tools/bpf/bpftool/cfg.c b/tools/bpf/bpftool/cfg.c index f30b3a4a840b..31f0db41513f 100644 --- a/tools/bpf/bpftool/cfg.c +++ b/tools/bpf/bpftool/cfg.c @@ -1,39 +1,5 @@  // SPDX-License-Identifier: (GPL-2.0-only OR BSD-2-Clause) -/* - * Copyright (C) 2018 Netronome Systems, Inc. - * - * This software is dual licensed under the GNU General License Version 2, - * June 1991 as shown in the file COPYING in the top-level directory of this - * source tree or the BSD 2-Clause License provided below.  You have the - * option to license this software under the complete terms of either license. - * - * The BSD 2-Clause License: - * - *     Redistribution and use in source and binary forms, with or - *     without modification, are permitted provided that the following - *     conditions are met: - * - *      1. Redistributions of source code must retain the above - *         copyright notice, this list of conditions and the following - *         disclaimer. - * - *      2. Redistributions in binary form must reproduce the above - *         copyright notice, this list of conditions and the following - *         disclaimer in the documentation and/or other materials - *         provided with the distribution. - * - * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" - * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE - * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE - * ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE - * LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR - * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF - * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS - * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN - * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) - * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE - * POSSIBILITY OF SUCH DAMAGE. - */ +/* Copyright (C) 2018 Netronome Systems, Inc. */  #include <linux/list.h>  #include <stdlib.h> diff --git a/tools/bpf/bpftool/cfg.h b/tools/bpf/bpftool/cfg.h index 2cc9bd990b13..e144257ea6d2 100644 --- a/tools/bpf/bpftool/cfg.h +++ b/tools/bpf/bpftool/cfg.h @@ -1,39 +1,5 @@ -// SPDX-License-Identifier: (GPL-2.0-only OR BSD-2-Clause) -/* - * Copyright (C) 2018 Netronome Systems, Inc. - * - * This software is dual licensed under the GNU General License Version 2, - * June 1991 as shown in the file COPYING in the top-level directory of this - * source tree or the BSD 2-Clause License provided below.  You have the - * option to license this software under the complete terms of either license. - * - * The BSD 2-Clause License: - * - *     Redistribution and use in source and binary forms, with or - *     without modification, are permitted provided that the following - *     conditions are met: - * - *      1. Redistributions of source code must retain the above - *         copyright notice, this list of conditions and the following - *         disclaimer. - * - *      2. Redistributions in binary form must reproduce the above - *         copyright notice, this list of conditions and the following - *         disclaimer in the documentation and/or other materials - *         provided with the distribution. - * - * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" - * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE - * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE - * ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE - * LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR - * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF - * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS - * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN - * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) - * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE - * POSSIBILITY OF SUCH DAMAGE. - */ +/* SPDX-License-Identifier: (GPL-2.0-only OR BSD-2-Clause) */ +/* Copyright (C) 2018 Netronome Systems, Inc. */  #ifndef __BPF_TOOL_CFG_H  #define __BPF_TOOL_CFG_H diff --git a/tools/bpf/bpftool/cgroup.c b/tools/bpf/bpftool/cgroup.c index ee7a9765c6b3..4b5c8da2a7c0 100644 --- a/tools/bpf/bpftool/cgroup.c +++ b/tools/bpf/bpftool/cgroup.c @@ -1,4 +1,4 @@ -// SPDX-License-Identifier: GPL-2.0+ +// SPDX-License-Identifier: (GPL-2.0-only OR BSD-2-Clause)  // Copyright (C) 2017 Facebook  // Author: Roman Gushchin <guro@fb.com> diff --git a/tools/bpf/bpftool/common.c b/tools/bpf/bpftool/common.c index 70fd48d79f61..897483457bf0 100644 --- a/tools/bpf/bpftool/common.c +++ b/tools/bpf/bpftool/common.c @@ -1,35 +1,5 @@ -/* - * Copyright (C) 2017-2018 Netronome Systems, Inc. - * - * This software is dual licensed under the GNU General License Version 2, - * June 1991 as shown in the file COPYING in the top-level directory of this - * source tree or the BSD 2-Clause License provided below.  You have the - * option to license this software under the complete terms of either license. - * - * The BSD 2-Clause License: - * - *     Redistribution and use in source and binary forms, with or - *     without modification, are permitted provided that the following - *     conditions are met: - * - *      1. Redistributions of source code must retain the above - *         copyright notice, this list of conditions and the following - *         disclaimer. - * - *      2. Redistributions in binary form must reproduce the above - *         copyright notice, this list of conditions and the following - *         disclaimer in the documentation and/or other materials - *         provided with the distribution. - * - * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, - * EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF - * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND - * NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS - * BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN - * ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN - * CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE - * SOFTWARE. - */ +// SPDX-License-Identifier: (GPL-2.0-only OR BSD-2-Clause) +/* Copyright (C) 2017-2018 Netronome Systems, Inc. */  #include <ctype.h>  #include <errno.h> @@ -46,8 +16,8 @@  #include <linux/magic.h>  #include <net/if.h>  #include <sys/mount.h> +#include <sys/resource.h>  #include <sys/stat.h> -#include <sys/types.h>  #include <sys/vfs.h>  #include <bpf.h> @@ -58,7 +28,7 @@  #define BPF_FS_MAGIC		0xcafe4a11  #endif -void p_err(const char *fmt, ...) +void __printf(1, 2) p_err(const char *fmt, ...)  {  	va_list ap; @@ -76,7 +46,7 @@ void p_err(const char *fmt, ...)  	va_end(ap);  } -void p_info(const char *fmt, ...) +void __printf(1, 2) p_info(const char *fmt, ...)  {  	va_list ap; @@ -99,7 +69,15 @@ static bool is_bpffs(char *path)  	return (unsigned long)st_fs.f_type == BPF_FS_MAGIC;  } -static int mnt_bpffs(const char *target, char *buff, size_t bufflen) +void set_max_rlimit(void) +{ +	struct rlimit rinf = { RLIM_INFINITY, RLIM_INFINITY }; + +	setrlimit(RLIMIT_MEMLOCK, &rinf); +} + +static int +mnt_fs(const char *target, const char *type, char *buff, size_t bufflen)  {  	bool bind_done = false; @@ -121,15 +99,29 @@ static int mnt_bpffs(const char *target, char *buff, size_t bufflen)  		bind_done = true;  	} -	if (mount("bpf", target, "bpf", 0, "mode=0700")) { -		snprintf(buff, bufflen, "mount -t bpf bpf %s failed: %s", -			 target, strerror(errno)); +	if (mount(type, target, type, 0, "mode=0700")) { +		snprintf(buff, bufflen, "mount -t %s %s %s failed: %s", +			 type, type, target, strerror(errno));  		return -1;  	}  	return 0;  } +int mount_tracefs(const char *target) +{ +	char err_str[ERR_MAX_LEN]; +	int err; + +	err = mnt_fs(target, "tracefs", err_str, ERR_MAX_LEN); +	if (err) { +		err_str[ERR_MAX_LEN - 1] = '\0'; +		p_err("can't mount tracefs: %s", err_str); +	} + +	return err; +} +  int open_obj_pinned(char *path, bool quiet)  {  	int fd; @@ -170,34 +162,29 @@ int open_obj_pinned_any(char *path, enum bpf_obj_type exp_type)  	return fd;  } -int do_pin_fd(int fd, const char *name) +int mount_bpffs_for_pin(const char *name)  {  	char err_str[ERR_MAX_LEN];  	char *file;  	char *dir;  	int err = 0; -	err = bpf_obj_pin(fd, name); -	if (!err) -		goto out; -  	file = malloc(strlen(name) + 1);  	strcpy(file, name);  	dir = dirname(file); -	if (errno != EPERM || is_bpffs(dir)) { -		p_err("can't pin the object (%s): %s", name, strerror(errno)); +	if (is_bpffs(dir)) +		/* nothing to do if already mounted */ +		goto out_free; + +	if (block_mount) { +		p_err("no BPF file system found, not mounting it due to --nomount option"); +		err = -1;  		goto out_free;  	} -	/* Attempt to mount bpffs, then retry pinning. */ -	err = mnt_bpffs(dir, err_str, ERR_MAX_LEN); -	if (!err) { -		err = bpf_obj_pin(fd, name); -		if (err) -			p_err("can't pin the object (%s): %s", name, -			      strerror(errno)); -	} else { +	err = mnt_fs(dir, "bpf", err_str, ERR_MAX_LEN); +	if (err) {  		err_str[ERR_MAX_LEN - 1] = '\0';  		p_err("can't mount BPF file system to pin the object (%s): %s",  		      name, err_str); @@ -205,10 +192,20 @@ int do_pin_fd(int fd, const char *name)  out_free:  	free(file); -out:  	return err;  } +int do_pin_fd(int fd, const char *name) +{ +	int err; + +	err = mount_bpffs_for_pin(name); +	if (err) +		return err; + +	return bpf_obj_pin(fd, name); +} +  int do_pin_any(int argc, char **argv, int (*get_fd_by_id)(__u32))  {  	unsigned int id; @@ -269,7 +266,7 @@ int get_fd_type(int fd)  	char buf[512];  	ssize_t n; -	snprintf(path, sizeof(path), "/proc/%d/fd/%d", getpid(), fd); +	snprintf(path, sizeof(path), "/proc/self/fd/%d", fd);  	n = readlink(path, buf, sizeof(buf));  	if (n < 0) { @@ -297,7 +294,7 @@ char *get_fdinfo(int fd, const char *key)  	ssize_t n;  	FILE *fdi; -	snprintf(path, sizeof(path), "/proc/%d/fdinfo/%d", getpid(), fd); +	snprintf(path, sizeof(path), "/proc/self/fdinfo/%d", fd);  	fdi = fopen(path, "r");  	if (!fdi) { @@ -598,7 +595,7 @@ void print_dev_plain(__u32 ifindex, __u64 ns_dev, __u64 ns_inode)  	if (!ifindex)  		return; -	printf(" dev "); +	printf("  offloaded_to ");  	if (ifindex_to_name_ns(ifindex, ns_dev, ns_inode, name))  		printf("%s", name);  	else diff --git a/tools/bpf/bpftool/jit_disasm.c b/tools/bpf/bpftool/jit_disasm.c index c75ffd9ce2bb..3ef3093560ba 100644 --- a/tools/bpf/bpftool/jit_disasm.c +++ b/tools/bpf/bpftool/jit_disasm.c @@ -1,3 +1,4 @@ +// SPDX-License-Identifier: (GPL-2.0-only OR BSD-2-Clause)  /*   * Based on:   * @@ -19,29 +20,21 @@  #include <string.h>  #include <bfd.h>  #include <dis-asm.h> -#include <sys/types.h>  #include <sys/stat.h>  #include <limits.h> +#include <libbpf.h>  #include "json_writer.h"  #include "main.h"  static void get_exec_path(char *tpath, size_t size)  { +	const char *path = "/proc/self/exe";  	ssize_t len; -	char *path; - -	snprintf(tpath, size, "/proc/%d/exe", (int) getpid()); -	tpath[size - 1] = 0; - -	path = strdup(tpath); -	assert(path);  	len = readlink(path, tpath, size - 1);  	assert(len > 0);  	tpath[len] = 0; - -	free(path);  }  static int oper_count; @@ -77,10 +70,16 @@ static int fprintf_json(void *out, const char *fmt, ...)  }  void disasm_print_insn(unsigned char *image, ssize_t len, int opcodes, -		       const char *arch, const char *disassembler_options) +		       const char *arch, const char *disassembler_options, +		       const struct btf *btf, +		       const struct bpf_prog_linfo *prog_linfo, +		       __u64 func_ksym, unsigned int func_idx, +		       bool linum)  { +	const struct bpf_line_info *linfo = NULL;  	disassembler_ftype disassemble;  	struct disassemble_info info; +	unsigned int nr_skip = 0;  	int count, i, pc = 0;  	char tpath[PATH_MAX];  	bfd *bfdf; @@ -109,7 +108,7 @@ void disasm_print_insn(unsigned char *image, ssize_t len, int opcodes,  		if (inf) {  			bfdf->arch_info = inf;  		} else { -			p_err("No libfd support for %s", arch); +			p_err("No libbfd support for %s", arch);  			return;  		}  	} @@ -136,12 +135,26 @@ void disasm_print_insn(unsigned char *image, ssize_t len, int opcodes,  	if (json_output)  		jsonw_start_array(json_wtr);  	do { +		if (prog_linfo) { +			linfo = bpf_prog_linfo__lfind_addr_func(prog_linfo, +								func_ksym + pc, +								func_idx, +								nr_skip); +			if (linfo) +				nr_skip++; +		} +  		if (json_output) {  			jsonw_start_object(json_wtr);  			oper_count = 0; +			if (linfo) +				btf_dump_linfo_json(btf, linfo, linum);  			jsonw_name(json_wtr, "pc");  			jsonw_printf(json_wtr, "\"0x%x\"", pc);  		} else { +			if (linfo) +				btf_dump_linfo_plain(btf, linfo, "; ", +						     linum);  			printf("%4x:\t", pc);  		} @@ -183,3 +196,9 @@ void disasm_print_insn(unsigned char *image, ssize_t len, int opcodes,  	bfd_close(bfdf);  } + +int disasm_init(void) +{ +	bfd_init(); +	return 0; +} diff --git a/tools/bpf/bpftool/json_writer.c b/tools/bpf/bpftool/json_writer.c index c6eef76322ae..bff7ee026680 100644 --- a/tools/bpf/bpftool/json_writer.c +++ b/tools/bpf/bpftool/json_writer.c @@ -1,3 +1,4 @@ +// SPDX-License-Identifier: (GPL-2.0-only OR BSD-2-Clause)  /*   * Simple streaming JSON writer   * @@ -19,6 +20,7 @@  #include <malloc.h>  #include <inttypes.h>  #include <stdint.h> +#include <linux/compiler.h>  #include "json_writer.h" @@ -156,7 +158,8 @@ void jsonw_name(json_writer_t *self, const char *name)  		putc(' ', self->out);  } -void jsonw_vprintf_enquote(json_writer_t *self, const char *fmt, va_list ap) +void __printf(2, 0) +jsonw_vprintf_enquote(json_writer_t *self, const char *fmt, va_list ap)  {  	jsonw_eor(self);  	putc('"', self->out); @@ -164,7 +167,7 @@ void jsonw_vprintf_enquote(json_writer_t *self, const char *fmt, va_list ap)  	putc('"', self->out);  } -void jsonw_printf(json_writer_t *self, const char *fmt, ...) +void __printf(2, 3) jsonw_printf(json_writer_t *self, const char *fmt, ...)  {  	va_list ap; diff --git a/tools/bpf/bpftool/json_writer.h b/tools/bpf/bpftool/json_writer.h index 0fa2fb1b6351..c1ab51aed99c 100644 --- a/tools/bpf/bpftool/json_writer.h +++ b/tools/bpf/bpftool/json_writer.h @@ -1,3 +1,4 @@ +/* SPDX-License-Identifier: (GPL-2.0-only OR BSD-2-Clause) */  /*   * Simple streaming JSON writer   * diff --git a/tools/bpf/bpftool/main.c b/tools/bpf/bpftool/main.c index 75a3296dc0bc..f44a1c2c4ea0 100644 --- a/tools/bpf/bpftool/main.c +++ b/tools/bpf/bpftool/main.c @@ -1,37 +1,6 @@ -/* - * Copyright (C) 2017-2018 Netronome Systems, Inc. - * - * This software is dual licensed under the GNU General License Version 2, - * June 1991 as shown in the file COPYING in the top-level directory of this - * source tree or the BSD 2-Clause License provided below.  You have the - * option to license this software under the complete terms of either license. - * - * The BSD 2-Clause License: - * - *     Redistribution and use in source and binary forms, with or - *     without modification, are permitted provided that the following - *     conditions are met: - * - *      1. Redistributions of source code must retain the above - *         copyright notice, this list of conditions and the following - *         disclaimer. - * - *      2. Redistributions in binary form must reproduce the above - *         copyright notice, this list of conditions and the following - *         disclaimer in the documentation and/or other materials - *         provided with the distribution. - * - * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, - * EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF - * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND - * NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS - * BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN - * ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN - * CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE - * SOFTWARE. - */ - -#include <bfd.h> +// SPDX-License-Identifier: (GPL-2.0-only OR BSD-2-Clause) +/* Copyright (C) 2017-2018 Netronome Systems, Inc. */ +  #include <ctype.h>  #include <errno.h>  #include <getopt.h> @@ -55,6 +24,7 @@ json_writer_t *json_wtr;  bool pretty_output;  bool json_output;  bool show_pinned; +bool block_mount;  int bpf_flags;  struct pinned_obj_table prog_table;  struct pinned_obj_table map_table; @@ -344,6 +314,7 @@ int main(int argc, char **argv)  		{ "version",	no_argument,	NULL,	'V' },  		{ "bpffs",	no_argument,	NULL,	'f' },  		{ "mapcompat",	no_argument,	NULL,	'm' }, +		{ "nomount",	no_argument,	NULL,	'n' },  		{ 0 }  	};  	int opt, ret; @@ -352,13 +323,14 @@ int main(int argc, char **argv)  	pretty_output = false;  	json_output = false;  	show_pinned = false; +	block_mount = false;  	bin_name = argv[0];  	hash_init(prog_table.table);  	hash_init(map_table.table);  	opterr = 0; -	while ((opt = getopt_long(argc, argv, "Vhpjfm", +	while ((opt = getopt_long(argc, argv, "Vhpjfmn",  				  options, NULL)) >= 0) {  		switch (opt) {  		case 'V': @@ -385,6 +357,9 @@ int main(int argc, char **argv)  		case 'm':  			bpf_flags = MAPS_RELAX_COMPAT;  			break; +		case 'n': +			block_mount = true; +			break;  		default:  			p_err("unrecognized option '%s'", argv[optind - 1]);  			if (json_output) @@ -399,8 +374,6 @@ int main(int argc, char **argv)  	if (argc < 0)  		usage(); -	bfd_init(); -  	ret = cmd_select(cmds, argc, argv, do_help);  	if (json_output) diff --git a/tools/bpf/bpftool/main.h b/tools/bpf/bpftool/main.h index a8bf1e2d9818..052c91d4dc55 100644 --- a/tools/bpf/bpftool/main.h +++ b/tools/bpf/bpftool/main.h @@ -1,35 +1,5 @@ -/* - * Copyright (C) 2017-2018 Netronome Systems, Inc. - * - * This software is dual licensed under the GNU General License Version 2, - * June 1991 as shown in the file COPYING in the top-level directory of this - * source tree or the BSD 2-Clause License provided below.  You have the - * option to license this software under the complete terms of either license. - * - * The BSD 2-Clause License: - * - *     Redistribution and use in source and binary forms, with or - *     without modification, are permitted provided that the following - *     conditions are met: - * - *      1. Redistributions of source code must retain the above - *         copyright notice, this list of conditions and the following - *         disclaimer. - * - *      2. Redistributions in binary form must reproduce the above - *         copyright notice, this list of conditions and the following - *         disclaimer in the documentation and/or other materials - *         provided with the distribution. - * - * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, - * EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF - * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND - * NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS - * BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN - * ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN - * CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE - * SOFTWARE. - */ +/* SPDX-License-Identifier: (GPL-2.0-only OR BSD-2-Clause) */ +/* Copyright (C) 2017-2018 Netronome Systems, Inc. */  #ifndef __BPF_TOOL_H  #define __BPF_TOOL_H @@ -74,10 +44,37 @@  #define HELP_SPEC_PROGRAM						\  	"PROG := { id PROG_ID | pinned FILE | tag PROG_TAG }"  #define HELP_SPEC_OPTIONS						\ -	"OPTIONS := { {-j|--json} [{-p|--pretty}] | {-f|--bpffs} | {-m|--mapcompat}" +	"OPTIONS := { {-j|--json} [{-p|--pretty}] | {-f|--bpffs} |\n"	\ +	"\t            {-m|--mapcompat} | {-n|--nomount} }"  #define HELP_SPEC_MAP							\  	"MAP := { id MAP_ID | pinned FILE }" +static const char * const prog_type_name[] = { +	[BPF_PROG_TYPE_UNSPEC]			= "unspec", +	[BPF_PROG_TYPE_SOCKET_FILTER]		= "socket_filter", +	[BPF_PROG_TYPE_KPROBE]			= "kprobe", +	[BPF_PROG_TYPE_SCHED_CLS]		= "sched_cls", +	[BPF_PROG_TYPE_SCHED_ACT]		= "sched_act", +	[BPF_PROG_TYPE_TRACEPOINT]		= "tracepoint", +	[BPF_PROG_TYPE_XDP]			= "xdp", +	[BPF_PROG_TYPE_PERF_EVENT]		= "perf_event", +	[BPF_PROG_TYPE_CGROUP_SKB]		= "cgroup_skb", +	[BPF_PROG_TYPE_CGROUP_SOCK]		= "cgroup_sock", +	[BPF_PROG_TYPE_LWT_IN]			= "lwt_in", +	[BPF_PROG_TYPE_LWT_OUT]			= "lwt_out", +	[BPF_PROG_TYPE_LWT_XMIT]		= "lwt_xmit", +	[BPF_PROG_TYPE_SOCK_OPS]		= "sock_ops", +	[BPF_PROG_TYPE_SK_SKB]			= "sk_skb", +	[BPF_PROG_TYPE_CGROUP_DEVICE]		= "cgroup_device", +	[BPF_PROG_TYPE_SK_MSG]			= "sk_msg", +	[BPF_PROG_TYPE_RAW_TRACEPOINT]		= "raw_tracepoint", +	[BPF_PROG_TYPE_CGROUP_SOCK_ADDR]	= "cgroup_sock_addr", +	[BPF_PROG_TYPE_LWT_SEG6LOCAL]		= "lwt_seg6local", +	[BPF_PROG_TYPE_LIRC_MODE2]		= "lirc_mode2", +	[BPF_PROG_TYPE_SK_REUSEPORT]		= "sk_reuseport", +	[BPF_PROG_TYPE_FLOW_DISSECTOR]		= "flow_dissector", +}; +  enum bpf_obj_type {  	BPF_OBJ_UNKNOWN,  	BPF_OBJ_PROG, @@ -89,6 +86,7 @@ extern const char *bin_name;  extern json_writer_t *json_wtr;  extern bool json_output;  extern bool show_pinned; +extern bool block_mount;  extern int bpf_flags;  extern struct pinned_obj_table prog_table;  extern struct pinned_obj_table map_table; @@ -100,6 +98,10 @@ bool is_prefix(const char *pfx, const char *str);  void fprint_hex(FILE *f, void *arg, unsigned int n, const char *sep);  void usage(void) __noreturn; +void set_max_rlimit(void); + +int mount_tracefs(const char *target); +  struct pinned_obj_table {  	DECLARE_HASHTABLE(table, 16);  }; @@ -110,6 +112,9 @@ struct pinned_obj {  	struct hlist_node hash;  }; +struct btf; +struct bpf_line_info; +  int build_pinned_obj_table(struct pinned_obj_table *table,  			   enum bpf_obj_type type);  void delete_pinned_obj_table(struct pinned_obj_table *tab); @@ -129,6 +134,7 @@ const char *get_fd_type_name(enum bpf_obj_type type);  char *get_fdinfo(int fd, const char *key);  int open_obj_pinned(char *path, bool quiet);  int open_obj_pinned_any(char *path, enum bpf_obj_type exp_type); +int mount_bpffs_for_pin(const char *name);  int do_pin_any(int argc, char **argv, int (*get_fd_by_id)(__u32));  int do_pin_fd(int fd, const char *name); @@ -138,14 +144,38 @@ int do_event_pipe(int argc, char **argv);  int do_cgroup(int argc, char **arg);  int do_perf(int argc, char **arg);  int do_net(int argc, char **arg); +int do_tracelog(int argc, char **arg);  int parse_u32_arg(int *argc, char ***argv, __u32 *val, const char *what);  int prog_parse_fd(int *argc, char ***argv);  int map_parse_fd(int *argc, char ***argv);  int map_parse_fd_and_info(int *argc, char ***argv, void *info, __u32 *info_len); +struct bpf_prog_linfo; +#ifdef HAVE_LIBBFD_SUPPORT +void disasm_print_insn(unsigned char *image, ssize_t len, int opcodes, +		       const char *arch, const char *disassembler_options, +		       const struct btf *btf, +		       const struct bpf_prog_linfo *prog_linfo, +		       __u64 func_ksym, unsigned int func_idx, +		       bool linum); +int disasm_init(void); +#else +static inline  void disasm_print_insn(unsigned char *image, ssize_t len, int opcodes, -		       const char *arch, const char *disassembler_options); +		       const char *arch, const char *disassembler_options, +		       const struct btf *btf, +		       const struct bpf_prog_linfo *prog_linfo, +		       __u64 func_ksym, unsigned int func_idx, +		       bool linum) +{ +} +static inline int disasm_init(void) +{ +	p_err("No libbfd support"); +	return -1; +} +#endif  void print_data_json(uint8_t *data, size_t len);  void print_hex_data_json(uint8_t *data, size_t len); @@ -170,6 +200,14 @@ struct btf_dumper {   */  int btf_dumper_type(const struct btf_dumper *d, __u32 type_id,  		    const void *data); +void btf_dumper_type_only(const struct btf *btf, __u32 func_type_id, +			  char *func_only, int size); + +void btf_dump_linfo_plain(const struct btf *btf, +			  const struct bpf_line_info *linfo, +			  const char *prefix, bool linum); +void btf_dump_linfo_json(const struct btf *btf, +			 const struct bpf_line_info *linfo, bool linum);  struct nlattr;  struct ifinfomsg; diff --git a/tools/bpf/bpftool/map.c b/tools/bpf/bpftool/map.c index 7bf38f0e152e..2037e3dc864b 100644 --- a/tools/bpf/bpftool/map.c +++ b/tools/bpf/bpftool/map.c @@ -1,35 +1,5 @@ -/* - * Copyright (C) 2017-2018 Netronome Systems, Inc. - * - * This software is dual licensed under the GNU General License Version 2, - * June 1991 as shown in the file COPYING in the top-level directory of this - * source tree or the BSD 2-Clause License provided below.  You have the - * option to license this software under the complete terms of either license. - * - * The BSD 2-Clause License: - * - *     Redistribution and use in source and binary forms, with or - *     without modification, are permitted provided that the following - *     conditions are met: - * - *      1. Redistributions of source code must retain the above - *         copyright notice, this list of conditions and the following - *         disclaimer. - * - *      2. Redistributions in binary form must reproduce the above - *         copyright notice, this list of conditions and the following - *         disclaimer in the documentation and/or other materials - *         provided with the distribution. - * - * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, - * EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF - * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND - * NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS - * BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN - * ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN - * CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE - * SOFTWARE. - */ +// SPDX-License-Identifier: (GPL-2.0-only OR BSD-2-Clause) +/* Copyright (C) 2017-2018 Netronome Systems, Inc. */  #include <assert.h>  #include <errno.h> @@ -52,28 +22,30 @@  #include "main.h"  static const char * const map_type_name[] = { -	[BPF_MAP_TYPE_UNSPEC]		= "unspec", -	[BPF_MAP_TYPE_HASH]		= "hash", -	[BPF_MAP_TYPE_ARRAY]		= "array", -	[BPF_MAP_TYPE_PROG_ARRAY]	= "prog_array", -	[BPF_MAP_TYPE_PERF_EVENT_ARRAY]	= "perf_event_array", -	[BPF_MAP_TYPE_PERCPU_HASH]	= "percpu_hash", -	[BPF_MAP_TYPE_PERCPU_ARRAY]	= "percpu_array", -	[BPF_MAP_TYPE_STACK_TRACE]	= "stack_trace", -	[BPF_MAP_TYPE_CGROUP_ARRAY]	= "cgroup_array", -	[BPF_MAP_TYPE_LRU_HASH]		= "lru_hash", -	[BPF_MAP_TYPE_LRU_PERCPU_HASH]	= "lru_percpu_hash", -	[BPF_MAP_TYPE_LPM_TRIE]		= "lpm_trie", -	[BPF_MAP_TYPE_ARRAY_OF_MAPS]	= "array_of_maps", -	[BPF_MAP_TYPE_HASH_OF_MAPS]	= "hash_of_maps", -	[BPF_MAP_TYPE_DEVMAP]		= "devmap", -	[BPF_MAP_TYPE_SOCKMAP]		= "sockmap", -	[BPF_MAP_TYPE_CPUMAP]		= "cpumap", -	[BPF_MAP_TYPE_XSKMAP]           = "xskmap", -	[BPF_MAP_TYPE_SOCKHASH]		= "sockhash", -	[BPF_MAP_TYPE_CGROUP_STORAGE]	= "cgroup_storage", -	[BPF_MAP_TYPE_REUSEPORT_SOCKARRAY] = "reuseport_sockarray", +	[BPF_MAP_TYPE_UNSPEC]			= "unspec", +	[BPF_MAP_TYPE_HASH]			= "hash", +	[BPF_MAP_TYPE_ARRAY]			= "array", +	[BPF_MAP_TYPE_PROG_ARRAY]		= "prog_array", +	[BPF_MAP_TYPE_PERF_EVENT_ARRAY]		= "perf_event_array", +	[BPF_MAP_TYPE_PERCPU_HASH]		= "percpu_hash", +	[BPF_MAP_TYPE_PERCPU_ARRAY]		= "percpu_array", +	[BPF_MAP_TYPE_STACK_TRACE]		= "stack_trace", +	[BPF_MAP_TYPE_CGROUP_ARRAY]		= "cgroup_array", +	[BPF_MAP_TYPE_LRU_HASH]			= "lru_hash", +	[BPF_MAP_TYPE_LRU_PERCPU_HASH]		= "lru_percpu_hash", +	[BPF_MAP_TYPE_LPM_TRIE]			= "lpm_trie", +	[BPF_MAP_TYPE_ARRAY_OF_MAPS]		= "array_of_maps", +	[BPF_MAP_TYPE_HASH_OF_MAPS]		= "hash_of_maps", +	[BPF_MAP_TYPE_DEVMAP]			= "devmap", +	[BPF_MAP_TYPE_SOCKMAP]			= "sockmap", +	[BPF_MAP_TYPE_CPUMAP]			= "cpumap", +	[BPF_MAP_TYPE_XSKMAP]			= "xskmap", +	[BPF_MAP_TYPE_SOCKHASH]			= "sockhash", +	[BPF_MAP_TYPE_CGROUP_STORAGE]		= "cgroup_storage", +	[BPF_MAP_TYPE_REUSEPORT_SOCKARRAY]	= "reuseport_sockarray",  	[BPF_MAP_TYPE_PERCPU_CGROUP_STORAGE]	= "percpu_cgroup_storage", +	[BPF_MAP_TYPE_QUEUE]			= "queue", +	[BPF_MAP_TYPE_STACK]			= "stack",  };  static bool map_is_per_cpu(__u32 type) @@ -215,70 +187,6 @@ err_end_obj:  	return ret;  } -static int get_btf(struct bpf_map_info *map_info, struct btf **btf) -{ -	struct bpf_btf_info btf_info = { 0 }; -	__u32 len = sizeof(btf_info); -	__u32 last_size; -	int btf_fd; -	void *ptr; -	int err; - -	err = 0; -	*btf = NULL; -	btf_fd = bpf_btf_get_fd_by_id(map_info->btf_id); -	if (btf_fd < 0) -		return 0; - -	/* we won't know btf_size until we call bpf_obj_get_info_by_fd(). so -	 * let's start with a sane default - 4KiB here - and resize it only if -	 * bpf_obj_get_info_by_fd() needs a bigger buffer. -	 */ -	btf_info.btf_size = 4096; -	last_size = btf_info.btf_size; -	ptr = malloc(last_size); -	if (!ptr) { -		err = -ENOMEM; -		goto exit_free; -	} - -	bzero(ptr, last_size); -	btf_info.btf = ptr_to_u64(ptr); -	err = bpf_obj_get_info_by_fd(btf_fd, &btf_info, &len); - -	if (!err && btf_info.btf_size > last_size) { -		void *temp_ptr; - -		last_size = btf_info.btf_size; -		temp_ptr = realloc(ptr, last_size); -		if (!temp_ptr) { -			err = -ENOMEM; -			goto exit_free; -		} -		ptr = temp_ptr; -		bzero(ptr, last_size); -		btf_info.btf = ptr_to_u64(ptr); -		err = bpf_obj_get_info_by_fd(btf_fd, &btf_info, &len); -	} - -	if (err || btf_info.btf_size > last_size) { -		err = errno; -		goto exit_free; -	} - -	*btf = btf__new((__u8 *)btf_info.btf, btf_info.btf_size, NULL); -	if (IS_ERR(*btf)) { -		err = PTR_ERR(*btf); -		*btf = NULL; -	} - -exit_free: -	close(btf_fd); -	free(ptr); - -	return err; -} -  static json_writer_t *get_btf_writer(void)  {  	json_writer_t *jw = jsonw_new(stdout); @@ -383,7 +291,10 @@ static void print_entry_plain(struct bpf_map_info *info, unsigned char *key,  		printf(single_line ? "  " : "\n");  		printf("value:%c", break_names ? '\n' : ' '); -		fprint_hex(stdout, value, info->value_size, " "); +		if (value) +			fprint_hex(stdout, value, info->value_size, " "); +		else +			printf("<no entry>");  		printf("\n");  	} else { @@ -398,8 +309,11 @@ static void print_entry_plain(struct bpf_map_info *info, unsigned char *key,  		for (i = 0; i < n; i++) {  			printf("value (CPU %02d):%c",  			       i, info->value_size > 16 ? '\n' : ' '); -			fprint_hex(stdout, value + i * step, -				   info->value_size, " "); +			if (value) +				fprint_hex(stdout, value + i * step, +					   info->value_size, " "); +			else +				printf("<no entry>");  			printf("\n");  		}  	} @@ -543,7 +457,6 @@ static int show_map_close_json(int fd, struct bpf_map_info *info)  	char *memlock;  	memlock = get_fdinfo(fd, "memlock"); -	close(fd);  	jsonw_start_object(json_wtr); @@ -570,6 +483,30 @@ static int show_map_close_json(int fd, struct bpf_map_info *info)  		jsonw_int_field(json_wtr, "bytes_memlock", atoi(memlock));  	free(memlock); +	if (info->type == BPF_MAP_TYPE_PROG_ARRAY) { +		char *owner_prog_type = get_fdinfo(fd, "owner_prog_type"); +		char *owner_jited = get_fdinfo(fd, "owner_jited"); + +		if (owner_prog_type) { +			unsigned int prog_type = atoi(owner_prog_type); + +			if (prog_type < ARRAY_SIZE(prog_type_name)) +				jsonw_string_field(json_wtr, "owner_prog_type", +						   prog_type_name[prog_type]); +			else +				jsonw_uint_field(json_wtr, "owner_prog_type", +						 prog_type); +		} +		if (atoi(owner_jited)) +			jsonw_bool_field(json_wtr, "owner_jited", true); +		else +			jsonw_bool_field(json_wtr, "owner_jited", false); + +		free(owner_prog_type); +		free(owner_jited); +	} +	close(fd); +  	if (!hash_empty(map_table.table)) {  		struct pinned_obj *obj; @@ -592,7 +529,6 @@ static int show_map_close_plain(int fd, struct bpf_map_info *info)  	char *memlock;  	memlock = get_fdinfo(fd, "memlock"); -	close(fd);  	printf("%u: ", info->id);  	if (info->type < ARRAY_SIZE(map_type_name)) @@ -613,6 +549,30 @@ static int show_map_close_plain(int fd, struct bpf_map_info *info)  		printf("  memlock %sB", memlock);  	free(memlock); +	if (info->type == BPF_MAP_TYPE_PROG_ARRAY) { +		char *owner_prog_type = get_fdinfo(fd, "owner_prog_type"); +		char *owner_jited = get_fdinfo(fd, "owner_jited"); + +		printf("\n\t"); +		if (owner_prog_type) { +			unsigned int prog_type = atoi(owner_prog_type); + +			if (prog_type < ARRAY_SIZE(prog_type_name)) +				printf("owner_prog_type %s  ", +				       prog_type_name[prog_type]); +			else +				printf("owner_prog_type %d  ", prog_type); +		} +		if (atoi(owner_jited)) +			printf("owner jited"); +		else +			printf("owner not jited"); + +		free(owner_prog_type); +		free(owner_jited); +	} +	close(fd); +  	printf("\n");  	if (!hash_empty(map_table.table)) {  		struct pinned_obj *obj; @@ -731,7 +691,11 @@ static int dump_map_elem(int fd, void *key, void *value,  		jsonw_string_field(json_wtr, "error", strerror(lookup_errno));  		jsonw_end_object(json_wtr);  	} else { -		print_entry_error(map_info, key, strerror(lookup_errno)); +		if (errno == ENOENT) +			print_entry_plain(map_info, key, NULL); +		else +			print_entry_error(map_info, key, +					  strerror(lookup_errno));  	}  	return 0; @@ -765,7 +729,7 @@ static int do_dump(int argc, char **argv)  	prev_key = NULL; -	err = get_btf(&info, &btf); +	err = btf__get_from_id(info.btf_id, &btf);  	if (err) {  		p_err("failed to get btf");  		goto exit_free; @@ -909,7 +873,7 @@ static int do_lookup(int argc, char **argv)  	}  	/* here means bpf_map_lookup_elem() succeeded */ -	err = get_btf(&info, &btf); +	err = btf__get_from_id(info.btf_id, &btf);  	if (err) {  		p_err("failed to get btf");  		goto exit_free; @@ -1140,6 +1104,8 @@ static int do_create(int argc, char **argv)  		return -1;  	} +	set_max_rlimit(); +  	fd = bpf_create_map_xattr(&attr);  	if (fd < 0) {  		p_err("map create failed: %s", strerror(errno)); diff --git a/tools/bpf/bpftool/map_perf_ring.c b/tools/bpf/bpftool/map_perf_ring.c index bdaf4062e26e..0507dfaf7a8f 100644 --- a/tools/bpf/bpftool/map_perf_ring.c +++ b/tools/bpf/bpftool/map_perf_ring.c @@ -1,4 +1,4 @@ -// SPDX-License-Identifier: GPL-2.0-only +// SPDX-License-Identifier: (GPL-2.0-only OR BSD-2-Clause)  /* Copyright (C) 2018 Netronome Systems, Inc. */  /* This program is free software; you can redistribute it and/or   * modify it under the terms of version 2 of the GNU General Public diff --git a/tools/bpf/bpftool/net.c b/tools/bpf/bpftool/net.c index d441bb7035ca..db0e7de49d49 100644 --- a/tools/bpf/bpftool/net.c +++ b/tools/bpf/bpftool/net.c @@ -1,4 +1,4 @@ -// SPDX-License-Identifier: GPL-2.0+ +// SPDX-License-Identifier: (GPL-2.0-only OR BSD-2-Clause)  // Copyright (C) 2018 Facebook  #define _GNU_SOURCE diff --git a/tools/bpf/bpftool/netlink_dumper.c b/tools/bpf/bpftool/netlink_dumper.c index 4e9f4531269f..550a0f537eed 100644 --- a/tools/bpf/bpftool/netlink_dumper.c +++ b/tools/bpf/bpftool/netlink_dumper.c @@ -1,4 +1,4 @@ -// SPDX-License-Identifier: GPL-2.0+ +// SPDX-License-Identifier: (GPL-2.0-only OR BSD-2-Clause)  // Copyright (C) 2018 Facebook  #include <stdlib.h> diff --git a/tools/bpf/bpftool/netlink_dumper.h b/tools/bpf/bpftool/netlink_dumper.h index e3516b586a34..774af6c62ef5 100644 --- a/tools/bpf/bpftool/netlink_dumper.h +++ b/tools/bpf/bpftool/netlink_dumper.h @@ -1,4 +1,4 @@ -// SPDX-License-Identifier: GPL-2.0+ +/* SPDX-License-Identifier: (GPL-2.0-only OR BSD-2-Clause) */  // Copyright (C) 2018 Facebook  #ifndef _NETLINK_DUMPER_H_ diff --git a/tools/bpf/bpftool/perf.c b/tools/bpf/bpftool/perf.c index b76b77dcfd1f..f2a545e667c4 100644 --- a/tools/bpf/bpftool/perf.c +++ b/tools/bpf/bpftool/perf.c @@ -1,4 +1,4 @@ -// SPDX-License-Identifier: GPL-2.0+ +// SPDX-License-Identifier: (GPL-2.0-only OR BSD-2-Clause)  // Copyright (C) 2018 Facebook  // Author: Yonghong Song <yhs@fb.com> diff --git a/tools/bpf/bpftool/prog.c b/tools/bpf/bpftool/prog.c index ccee180dfb76..2d1bb7d6ff51 100644 --- a/tools/bpf/bpftool/prog.c +++ b/tools/bpf/bpftool/prog.c @@ -1,35 +1,5 @@ -/* - * Copyright (C) 2017-2018 Netronome Systems, Inc. - * - * This software is dual licensed under the GNU General License Version 2, - * June 1991 as shown in the file COPYING in the top-level directory of this - * source tree or the BSD 2-Clause License provided below.  You have the - * option to license this software under the complete terms of either license. - * - * The BSD 2-Clause License: - * - *     Redistribution and use in source and binary forms, with or - *     without modification, are permitted provided that the following - *     conditions are met: - * - *      1. Redistributions of source code must retain the above - *         copyright notice, this list of conditions and the following - *         disclaimer. - * - *      2. Redistributions in binary form must reproduce the above - *         copyright notice, this list of conditions and the following - *         disclaimer in the documentation and/or other materials - *         provided with the distribution. - * - * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, - * EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF - * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND - * NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS - * BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN - * ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN - * CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE - * SOFTWARE. - */ +// SPDX-License-Identifier: (GPL-2.0-only OR BSD-2-Clause) +/* Copyright (C) 2017-2018 Netronome Systems, Inc. */  #define _GNU_SOURCE  #include <errno.h> @@ -47,44 +17,22 @@  #include <linux/err.h>  #include <bpf.h> +#include <btf.h>  #include <libbpf.h>  #include "cfg.h"  #include "main.h"  #include "xlated_dumper.h" -static const char * const prog_type_name[] = { -	[BPF_PROG_TYPE_UNSPEC]		= "unspec", -	[BPF_PROG_TYPE_SOCKET_FILTER]	= "socket_filter", -	[BPF_PROG_TYPE_KPROBE]		= "kprobe", -	[BPF_PROG_TYPE_SCHED_CLS]	= "sched_cls", -	[BPF_PROG_TYPE_SCHED_ACT]	= "sched_act", -	[BPF_PROG_TYPE_TRACEPOINT]	= "tracepoint", -	[BPF_PROG_TYPE_XDP]		= "xdp", -	[BPF_PROG_TYPE_PERF_EVENT]	= "perf_event", -	[BPF_PROG_TYPE_CGROUP_SKB]	= "cgroup_skb", -	[BPF_PROG_TYPE_CGROUP_SOCK]	= "cgroup_sock", -	[BPF_PROG_TYPE_LWT_IN]		= "lwt_in", -	[BPF_PROG_TYPE_LWT_OUT]		= "lwt_out", -	[BPF_PROG_TYPE_LWT_XMIT]	= "lwt_xmit", -	[BPF_PROG_TYPE_SOCK_OPS]	= "sock_ops", -	[BPF_PROG_TYPE_SK_SKB]		= "sk_skb", -	[BPF_PROG_TYPE_CGROUP_DEVICE]	= "cgroup_device", -	[BPF_PROG_TYPE_SK_MSG]		= "sk_msg", -	[BPF_PROG_TYPE_RAW_TRACEPOINT]	= "raw_tracepoint", -	[BPF_PROG_TYPE_CGROUP_SOCK_ADDR] = "cgroup_sock_addr", -	[BPF_PROG_TYPE_LIRC_MODE2]	= "lirc_mode2", -	[BPF_PROG_TYPE_FLOW_DISSECTOR]	= "flow_dissector", -}; -  static const char * const attach_type_strings[] = {  	[BPF_SK_SKB_STREAM_PARSER] = "stream_parser",  	[BPF_SK_SKB_STREAM_VERDICT] = "stream_verdict",  	[BPF_SK_MSG_VERDICT] = "msg_verdict", +	[BPF_FLOW_DISSECTOR] = "flow_dissector",  	[__MAX_BPF_ATTACH_TYPE] = NULL,  }; -enum bpf_attach_type parse_attach_type(const char *str) +static enum bpf_attach_type parse_attach_type(const char *str)  {  	enum bpf_attach_type type; @@ -445,6 +393,10 @@ static int do_show(int argc, char **argv)  static int do_dump(int argc, char **argv)  { +	unsigned int finfo_rec_size, linfo_rec_size, jited_linfo_rec_size; +	void *func_info = NULL, *linfo = NULL, *jited_linfo = NULL; +	unsigned int nr_finfo, nr_linfo = 0, nr_jited_linfo = 0; +	struct bpf_prog_linfo *prog_linfo = NULL;  	unsigned long *func_ksyms = NULL;  	struct bpf_prog_info info = {};  	unsigned int *func_lens = NULL; @@ -453,11 +405,14 @@ static int do_dump(int argc, char **argv)  	unsigned int nr_func_lens;  	struct dump_data dd = {};  	__u32 len = sizeof(info); +	struct btf *btf = NULL;  	unsigned int buf_size;  	char *filepath = NULL;  	bool opcodes = false;  	bool visual = false; +	char func_sig[1024];  	unsigned char *buf; +	bool linum = false;  	__u32 *member_len;  	__u64 *member_ptr;  	ssize_t n; @@ -465,6 +420,9 @@ static int do_dump(int argc, char **argv)  	int fd;  	if (is_prefix(*argv, "jited")) { +		if (disasm_init()) +			return -1; +  		member_len = &info.jited_prog_len;  		member_ptr = &info.jited_prog_insns;  	} else if (is_prefix(*argv, "xlated")) { @@ -498,6 +456,9 @@ static int do_dump(int argc, char **argv)  	} else if (is_prefix(*argv, "visual")) {  		visual = true;  		NEXT_ARG(); +	} else if (is_prefix(*argv, "linum")) { +		linum = true; +		NEXT_ARG();  	}  	if (argc) { @@ -546,6 +507,43 @@ static int do_dump(int argc, char **argv)  		}  	} +	nr_finfo = info.nr_func_info; +	finfo_rec_size = info.func_info_rec_size; +	if (nr_finfo && finfo_rec_size) { +		func_info = malloc(nr_finfo * finfo_rec_size); +		if (!func_info) { +			p_err("mem alloc failed"); +			close(fd); +			goto err_free; +		} +	} + +	linfo_rec_size = info.line_info_rec_size; +	if (info.nr_line_info && linfo_rec_size && info.btf_id) { +		nr_linfo = info.nr_line_info; +		linfo = malloc(nr_linfo * linfo_rec_size); +		if (!linfo) { +			p_err("mem alloc failed"); +			close(fd); +			goto err_free; +		} +	} + +	jited_linfo_rec_size = info.jited_line_info_rec_size; +	if (info.nr_jited_line_info && +	    jited_linfo_rec_size && +	    info.nr_jited_ksyms && +	    info.nr_jited_func_lens && +	    info.btf_id) { +		nr_jited_linfo = info.nr_jited_line_info; +		jited_linfo = malloc(nr_jited_linfo * jited_linfo_rec_size); +		if (!jited_linfo) { +			p_err("mem alloc failed"); +			close(fd); +			goto err_free; +		} +	} +  	memset(&info, 0, sizeof(info));  	*member_ptr = ptr_to_u64(buf); @@ -554,6 +552,15 @@ static int do_dump(int argc, char **argv)  	info.nr_jited_ksyms = nr_func_ksyms;  	info.jited_func_lens = ptr_to_u64(func_lens);  	info.nr_jited_func_lens = nr_func_lens; +	info.nr_func_info = nr_finfo; +	info.func_info_rec_size = finfo_rec_size; +	info.func_info = ptr_to_u64(func_info); +	info.nr_line_info = nr_linfo; +	info.line_info_rec_size = linfo_rec_size; +	info.line_info = ptr_to_u64(linfo); +	info.nr_jited_line_info = nr_jited_linfo; +	info.jited_line_info_rec_size = jited_linfo_rec_size; +	info.jited_line_info = ptr_to_u64(jited_linfo);  	err = bpf_obj_get_info_by_fd(fd, &info, &len);  	close(fd); @@ -577,6 +584,42 @@ static int do_dump(int argc, char **argv)  		goto err_free;  	} +	if (info.nr_func_info != nr_finfo) { +		p_err("incorrect nr_func_info %d vs. expected %d", +		      info.nr_func_info, nr_finfo); +		goto err_free; +	} + +	if (info.func_info_rec_size != finfo_rec_size) { +		p_err("incorrect func_info_rec_size %d vs. expected %d", +		      info.func_info_rec_size, finfo_rec_size); +		goto err_free; +	} + +	if (linfo && info.nr_line_info != nr_linfo) { +		p_err("incorrect nr_line_info %u vs. expected %u", +		      info.nr_line_info, nr_linfo); +		goto err_free; +	} + +	if (info.line_info_rec_size != linfo_rec_size) { +		p_err("incorrect line_info_rec_size %u vs. expected %u", +		      info.line_info_rec_size, linfo_rec_size); +		goto err_free; +	} + +	if (jited_linfo && info.nr_jited_line_info != nr_jited_linfo) { +		p_err("incorrect nr_jited_line_info %u vs. expected %u", +		      info.nr_jited_line_info, nr_jited_linfo); +		goto err_free; +	} + +	if (info.jited_line_info_rec_size != jited_linfo_rec_size) { +		p_err("incorrect jited_line_info_rec_size %u vs. expected %u", +		      info.jited_line_info_rec_size, jited_linfo_rec_size); +		goto err_free; +	} +  	if ((member_len == &info.jited_prog_len &&  	     info.jited_prog_insns == 0) ||  	    (member_len == &info.xlated_prog_len && @@ -585,6 +628,17 @@ static int do_dump(int argc, char **argv)  		goto err_free;  	} +	if (info.btf_id && btf__get_from_id(info.btf_id, &btf)) { +		p_err("failed to get btf"); +		goto err_free; +	} + +	if (nr_linfo) { +		prog_linfo = bpf_prog_linfo__new(&info); +		if (!prog_linfo) +			p_info("error in processing bpf_line_info.  continue without it."); +	} +  	if (filepath) {  		fd = open(filepath, O_WRONLY | O_CREAT | O_TRUNC, 0600);  		if (fd < 0) { @@ -617,6 +671,7 @@ static int do_dump(int argc, char **argv)  		if (info.nr_jited_func_lens && info.jited_func_lens) {  			struct kernel_sym *sym = NULL; +			struct bpf_func_info *record;  			char sym_name[SYM_MAX_NAME];  			unsigned char *img = buf;  			__u64 *ksyms = NULL; @@ -643,17 +698,33 @@ static int do_dump(int argc, char **argv)  					strcpy(sym_name, "unknown");  				} +				if (func_info) { +					record = func_info + i * finfo_rec_size; +					btf_dumper_type_only(btf, record->type_id, +							     func_sig, +							     sizeof(func_sig)); +				} +  				if (json_output) {  					jsonw_start_object(json_wtr); +					if (func_info && func_sig[0] != '\0') { +						jsonw_name(json_wtr, "proto"); +						jsonw_string(json_wtr, func_sig); +					}  					jsonw_name(json_wtr, "name");  					jsonw_string(json_wtr, sym_name);  					jsonw_name(json_wtr, "insns");  				} else { +					if (func_info && func_sig[0] != '\0') +						printf("%s:\n", func_sig);  					printf("%s:\n", sym_name);  				} -				disasm_print_insn(img, lens[i], opcodes, name, -						  disasm_opt); +				disasm_print_insn(img, lens[i], opcodes, +						  name, disasm_opt, btf, +						  prog_linfo, ksyms[i], i, +						  linum); +  				img += lens[i];  				if (json_output) @@ -666,7 +737,7 @@ static int do_dump(int argc, char **argv)  				jsonw_end_array(json_wtr);  		} else {  			disasm_print_insn(buf, *member_len, opcodes, name, -					  disasm_opt); +					  disasm_opt, btf, NULL, 0, 0, false);  		}  	} else if (visual) {  		if (json_output) @@ -677,23 +748,37 @@ static int do_dump(int argc, char **argv)  		kernel_syms_load(&dd);  		dd.nr_jited_ksyms = info.nr_jited_ksyms;  		dd.jited_ksyms = (__u64 *) info.jited_ksyms; +		dd.btf = btf; +		dd.func_info = func_info; +		dd.finfo_rec_size = finfo_rec_size; +		dd.prog_linfo = prog_linfo;  		if (json_output) -			dump_xlated_json(&dd, buf, *member_len, opcodes); +			dump_xlated_json(&dd, buf, *member_len, opcodes, +					 linum);  		else -			dump_xlated_plain(&dd, buf, *member_len, opcodes); +			dump_xlated_plain(&dd, buf, *member_len, opcodes, +					  linum);  		kernel_syms_destroy(&dd);  	}  	free(buf);  	free(func_ksyms);  	free(func_lens); +	free(func_info); +	free(linfo); +	free(jited_linfo); +	bpf_prog_linfo__free(prog_linfo);  	return 0;  err_free:  	free(buf);  	free(func_ksyms);  	free(func_lens); +	free(func_info); +	free(linfo); +	free(jited_linfo); +	bpf_prog_linfo__free(prog_linfo);  	return -1;  } @@ -713,37 +798,56 @@ struct map_replace {  	char *name;  }; -int map_replace_compar(const void *p1, const void *p2) +static int map_replace_compar(const void *p1, const void *p2)  {  	const struct map_replace *a = p1, *b = p2;  	return a->idx - b->idx;  } -static int do_attach(int argc, char **argv) +static int parse_attach_detach_args(int argc, char **argv, int *progfd, +				    enum bpf_attach_type *attach_type, +				    int *mapfd)  { -	enum bpf_attach_type attach_type; -	int err, mapfd, progfd; - -	if (!REQ_ARGS(5)) { -		p_err("too few parameters for map attach"); +	if (!REQ_ARGS(3))  		return -EINVAL; -	} -	progfd = prog_parse_fd(&argc, &argv); -	if (progfd < 0) -		return progfd; +	*progfd = prog_parse_fd(&argc, &argv); +	if (*progfd < 0) +		return *progfd; -	attach_type = parse_attach_type(*argv); -	if (attach_type == __MAX_BPF_ATTACH_TYPE) { -		p_err("invalid attach type"); +	*attach_type = parse_attach_type(*argv); +	if (*attach_type == __MAX_BPF_ATTACH_TYPE) { +		p_err("invalid attach/detach type");  		return -EINVAL;  	} + +	if (*attach_type == BPF_FLOW_DISSECTOR) { +		*mapfd = -1; +		return 0; +	} +  	NEXT_ARG(); +	if (!REQ_ARGS(2)) +		return -EINVAL; + +	*mapfd = map_parse_fd(&argc, &argv); +	if (*mapfd < 0) +		return *mapfd; + +	return 0; +} -	mapfd = map_parse_fd(&argc, &argv); -	if (mapfd < 0) -		return mapfd; +static int do_attach(int argc, char **argv) +{ +	enum bpf_attach_type attach_type; +	int err, progfd; +	int mapfd; + +	err = parse_attach_detach_args(argc, argv, +				       &progfd, &attach_type, &mapfd); +	if (err) +		return err;  	err = bpf_prog_attach(progfd, mapfd, attach_type, 0);  	if (err) { @@ -759,27 +863,13 @@ static int do_attach(int argc, char **argv)  static int do_detach(int argc, char **argv)  {  	enum bpf_attach_type attach_type; -	int err, mapfd, progfd; - -	if (!REQ_ARGS(5)) { -		p_err("too few parameters for map detach"); -		return -EINVAL; -	} +	int err, progfd; +	int mapfd; -	progfd = prog_parse_fd(&argc, &argv); -	if (progfd < 0) -		return progfd; - -	attach_type = parse_attach_type(*argv); -	if (attach_type == __MAX_BPF_ATTACH_TYPE) { -		p_err("invalid attach type"); -		return -EINVAL; -	} -	NEXT_ARG(); - -	mapfd = map_parse_fd(&argc, &argv); -	if (mapfd < 0) -		return mapfd; +	err = parse_attach_detach_args(argc, argv, +				       &progfd, &attach_type, &mapfd); +	if (err) +		return err;  	err = bpf_prog_detach2(progfd, mapfd, attach_type);  	if (err) { @@ -791,15 +881,17 @@ static int do_detach(int argc, char **argv)  		jsonw_null(json_wtr);  	return 0;  } -static int do_load(int argc, char **argv) + +static int load_with_options(int argc, char **argv, bool first_prog_only)  {  	enum bpf_attach_type expected_attach_type;  	struct bpf_object_open_attr attr = {  		.prog_type	= BPF_PROG_TYPE_UNSPEC,  	};  	struct map_replace *map_replace = NULL; +	struct bpf_program *prog = NULL, *pos;  	unsigned int old_map_fds = 0; -	struct bpf_program *prog; +	const char *pinmaps = NULL;  	struct bpf_object *obj;  	struct bpf_map *map;  	const char *pinfile; @@ -908,6 +1000,13 @@ static int do_load(int argc, char **argv)  				goto err_free_reuse_maps;  			}  			NEXT_ARG(); +		} else if (is_prefix(*argv, "pinmaps")) { +			NEXT_ARG(); + +			if (!REQ_ARGS(1)) +				goto err_free_reuse_maps; + +			pinmaps = GET_ARG();  		} else {  			p_err("expected no more arguments, 'type', 'map' or 'dev', got: '%s'?",  			      *argv); @@ -921,26 +1020,25 @@ static int do_load(int argc, char **argv)  		goto err_free_reuse_maps;  	} -	prog = bpf_program__next(NULL, obj); -	if (!prog) { -		p_err("object file doesn't contain any bpf program"); -		goto err_close_obj; -	} +	bpf_object__for_each_program(pos, obj) { +		enum bpf_prog_type prog_type = attr.prog_type; -	bpf_program__set_ifindex(prog, ifindex); -	if (attr.prog_type == BPF_PROG_TYPE_UNSPEC) { -		const char *sec_name = bpf_program__title(prog, false); +		if (attr.prog_type == BPF_PROG_TYPE_UNSPEC) { +			const char *sec_name = bpf_program__title(pos, false); -		err = libbpf_prog_type_by_name(sec_name, &attr.prog_type, -					       &expected_attach_type); -		if (err < 0) { -			p_err("failed to guess program type based on section name %s\n", -			      sec_name); -			goto err_close_obj; +			err = libbpf_prog_type_by_name(sec_name, &prog_type, +						       &expected_attach_type); +			if (err < 0) { +				p_err("failed to guess program type based on section name %s\n", +				      sec_name); +				goto err_close_obj; +			}  		} + +		bpf_program__set_ifindex(pos, ifindex); +		bpf_program__set_type(pos, prog_type); +		bpf_program__set_expected_attach_type(pos, expected_attach_type);  	} -	bpf_program__set_type(prog, attr.prog_type); -	bpf_program__set_expected_attach_type(prog, expected_attach_type);  	qsort(map_replace, old_map_fds, sizeof(*map_replace),  	      map_replace_compar); @@ -998,15 +1096,47 @@ static int do_load(int argc, char **argv)  		goto err_close_obj;  	} +	set_max_rlimit(); +  	err = bpf_object__load(obj);  	if (err) {  		p_err("failed to load object file");  		goto err_close_obj;  	} -	if (do_pin_fd(bpf_program__fd(prog), pinfile)) +	err = mount_bpffs_for_pin(pinfile); +	if (err)  		goto err_close_obj; +	if (first_prog_only) { +		prog = bpf_program__next(NULL, obj); +		if (!prog) { +			p_err("object file doesn't contain any bpf program"); +			goto err_close_obj; +		} + +		err = bpf_obj_pin(bpf_program__fd(prog), pinfile); +		if (err) { +			p_err("failed to pin program %s", +			      bpf_program__title(prog, false)); +			goto err_close_obj; +		} +	} else { +		err = bpf_object__pin_programs(obj, pinfile); +		if (err) { +			p_err("failed to pin all programs"); +			goto err_close_obj; +		} +	} + +	if (pinmaps) { +		err = bpf_object__pin_maps(obj, pinmaps); +		if (err) { +			p_err("failed to pin all maps"); +			goto err_unpin; +		} +	} +  	if (json_output)  		jsonw_null(json_wtr); @@ -1017,6 +1147,11 @@ static int do_load(int argc, char **argv)  	return 0; +err_unpin: +	if (first_prog_only) +		unlink(pinfile); +	else +		bpf_object__unpin_programs(obj, pinfile);  err_close_obj:  	bpf_object__close(obj);  err_free_reuse_maps: @@ -1026,6 +1161,16 @@ err_free_reuse_maps:  	return -1;  } +static int do_load(int argc, char **argv) +{ +	return load_with_options(argc, argv, true); +} + +static int do_loadall(int argc, char **argv) +{ +	return load_with_options(argc, argv, false); +} +  static int do_help(int argc, char **argv)  {  	if (json_output) { @@ -1035,13 +1180,16 @@ static int do_help(int argc, char **argv)  	fprintf(stderr,  		"Usage: %s %s { show | list } [PROG]\n" -		"       %s %s dump xlated PROG [{ file FILE | opcodes | visual }]\n" -		"       %s %s dump jited  PROG [{ file FILE | opcodes }]\n" +		"       %s %s dump xlated PROG [{ file FILE | opcodes | visual | linum }]\n" +		"       %s %s dump jited  PROG [{ file FILE | opcodes | linum }]\n"  		"       %s %s pin   PROG FILE\n" -		"       %s %s load  OBJ  FILE [type TYPE] [dev NAME] \\\n" -		"                         [map { idx IDX | name NAME } MAP]\n" -		"       %s %s attach PROG ATTACH_TYPE MAP\n" -		"       %s %s detach PROG ATTACH_TYPE MAP\n" +		"       %s %s { load | loadall } OBJ  PATH \\\n" +		"                         [type TYPE] [dev NAME] \\\n" +		"                         [map { idx IDX | name NAME } MAP]\\\n" +		"                         [pinmaps MAP_DIR]\n" +		"       %s %s attach PROG ATTACH_TYPE [MAP]\n" +		"       %s %s detach PROG ATTACH_TYPE [MAP]\n" +		"       %s %s tracelog\n"  		"       %s %s help\n"  		"\n"  		"       " HELP_SPEC_MAP "\n" @@ -1050,15 +1198,17 @@ static int do_help(int argc, char **argv)  		"                 tracepoint | raw_tracepoint | xdp | perf_event | cgroup/skb |\n"  		"                 cgroup/sock | cgroup/dev | lwt_in | lwt_out | lwt_xmit |\n"  		"                 lwt_seg6local | sockops | sk_skb | sk_msg | lirc_mode2 |\n" +		"                 sk_reuseport | flow_dissector |\n"  		"                 cgroup/bind4 | cgroup/bind6 | cgroup/post_bind4 |\n"  		"                 cgroup/post_bind6 | cgroup/connect4 | cgroup/connect6 |\n"  		"                 cgroup/sendmsg4 | cgroup/sendmsg6 }\n" -		"       ATTACH_TYPE := { msg_verdict | skb_verdict | skb_parse }\n" +		"       ATTACH_TYPE := { msg_verdict | skb_verdict | skb_parse |\n" +		"                        flow_dissector }\n"  		"       " HELP_SPEC_OPTIONS "\n"  		"",  		bin_name, argv[-2], bin_name, argv[-2], bin_name, argv[-2],  		bin_name, argv[-2], bin_name, argv[-2], bin_name, argv[-2], -		bin_name, argv[-2], bin_name, argv[-2]); +		bin_name, argv[-2], bin_name, argv[-2], bin_name, argv[-2]);  	return 0;  } @@ -1070,8 +1220,10 @@ static const struct cmd cmds[] = {  	{ "dump",	do_dump },  	{ "pin",	do_pin },  	{ "load",	do_load }, +	{ "loadall",	do_loadall },  	{ "attach",	do_attach },  	{ "detach",	do_detach }, +	{ "tracelog",	do_tracelog },  	{ 0 }  }; diff --git a/tools/bpf/bpftool/tracelog.c b/tools/bpf/bpftool/tracelog.c new file mode 100644 index 000000000000..e80a5c79b38f --- /dev/null +++ b/tools/bpf/bpftool/tracelog.c @@ -0,0 +1,166 @@ +// SPDX-License-Identifier: (GPL-2.0-only OR BSD-2-Clause) +/* Copyright (c) 2015-2017 Daniel Borkmann */ +/* Copyright (c) 2018 Netronome Systems, Inc. */ + +#include <errno.h> +#include <limits.h> +#include <signal.h> +#include <stdio.h> +#include <string.h> +#include <unistd.h> +#include <linux/magic.h> +#include <sys/fcntl.h> +#include <sys/vfs.h> + +#include "main.h" + +#ifndef TRACEFS_MAGIC +# define TRACEFS_MAGIC	0x74726163 +#endif + +#define _textify(x)	#x +#define textify(x)	_textify(x) + +FILE *trace_pipe_fd; +char *buff; + +static int validate_tracefs_mnt(const char *mnt, unsigned long magic) +{ +	struct statfs st_fs; + +	if (statfs(mnt, &st_fs) < 0) +		return -ENOENT; +	if ((unsigned long)st_fs.f_type != magic) +		return -ENOENT; + +	return 0; +} + +static bool +find_tracefs_mnt_single(unsigned long magic, char *mnt, const char *mntpt) +{ +	size_t src_len; + +	if (validate_tracefs_mnt(mntpt, magic)) +		return false; + +	src_len = strlen(mntpt); +	if (src_len + 1 >= PATH_MAX) { +		p_err("tracefs mount point name too long"); +		return false; +	} + +	strcpy(mnt, mntpt); +	return true; +} + +static bool get_tracefs_pipe(char *mnt) +{ +	static const char * const known_mnts[] = { +		"/sys/kernel/debug/tracing", +		"/sys/kernel/tracing", +		"/tracing", +		"/trace", +	}; +	const char *pipe_name = "/trace_pipe"; +	const char *fstype = "tracefs"; +	char type[100], format[32]; +	const char * const *ptr; +	bool found = false; +	FILE *fp; + +	for (ptr = known_mnts; ptr < known_mnts + ARRAY_SIZE(known_mnts); ptr++) +		if (find_tracefs_mnt_single(TRACEFS_MAGIC, mnt, *ptr)) +			goto exit_found; + +	fp = fopen("/proc/mounts", "r"); +	if (!fp) +		return false; + +	/* Allow room for NULL terminating byte and pipe file name */ +	snprintf(format, sizeof(format), "%%*s %%%zds %%99s %%*s %%*d %%*d\\n", +		 PATH_MAX - strlen(pipe_name) - 1); +	while (fscanf(fp, format, mnt, type) == 2) +		if (strcmp(type, fstype) == 0) { +			found = true; +			break; +		} +	fclose(fp); + +	/* The string from fscanf() might be truncated, check mnt is valid */ +	if (found && validate_tracefs_mnt(mnt, TRACEFS_MAGIC)) +		goto exit_found; + +	if (block_mount) +		return false; + +	p_info("could not find tracefs, attempting to mount it now"); +	/* Most of the time, tracefs is automatically mounted by debugfs at +	 * /sys/kernel/debug/tracing when we try to access it. If we could not +	 * find it, it is likely that debugfs is not mounted. Let's give one +	 * attempt at mounting just tracefs at /sys/kernel/tracing. +	 */ +	strcpy(mnt, known_mnts[1]); +	if (mount_tracefs(mnt)) +		return false; + +exit_found: +	strcat(mnt, pipe_name); +	return true; +} + +static void exit_tracelog(int signum) +{ +	fclose(trace_pipe_fd); +	free(buff); + +	if (json_output) { +		jsonw_end_array(json_wtr); +		jsonw_destroy(&json_wtr); +	} + +	exit(0); +} + +int do_tracelog(int argc, char **argv) +{ +	const struct sigaction act = { +		.sa_handler = exit_tracelog +	}; +	char trace_pipe[PATH_MAX]; +	size_t buff_len = 0; + +	if (json_output) +		jsonw_start_array(json_wtr); + +	if (!get_tracefs_pipe(trace_pipe)) +		return -1; + +	trace_pipe_fd = fopen(trace_pipe, "r"); +	if (!trace_pipe_fd) { +		p_err("could not open trace pipe: %s", strerror(errno)); +		return -1; +	} + +	sigaction(SIGHUP, &act, NULL); +	sigaction(SIGINT, &act, NULL); +	sigaction(SIGTERM, &act, NULL); +	while (1) { +		ssize_t ret; + +		ret = getline(&buff, &buff_len, trace_pipe_fd); +		if (ret <= 0) { +			p_err("failed to read content from trace pipe: %s", +			      strerror(errno)); +			break; +		} +		if (json_output) +			jsonw_string(json_wtr, buff); +		else +			printf("%s", buff); +	} + +	fclose(trace_pipe_fd); +	free(buff); +	return -1; +} diff --git a/tools/bpf/bpftool/xlated_dumper.c b/tools/bpf/bpftool/xlated_dumper.c index 3284759df98a..7073dbe1ff27 100644 --- a/tools/bpf/bpftool/xlated_dumper.c +++ b/tools/bpf/bpftool/xlated_dumper.c @@ -1,39 +1,5 @@  // SPDX-License-Identifier: (GPL-2.0-only OR BSD-2-Clause) -/* - * Copyright (C) 2018 Netronome Systems, Inc. - * - * This software is dual licensed under the GNU General License Version 2, - * June 1991 as shown in the file COPYING in the top-level directory of this - * source tree or the BSD 2-Clause License provided below.  You have the - * option to license this software under the complete terms of either license. - * - * The BSD 2-Clause License: - * - *     Redistribution and use in source and binary forms, with or - *     without modification, are permitted provided that the following - *     conditions are met: - * - *      1. Redistributions of source code must retain the above - *         copyright notice, this list of conditions and the following - *         disclaimer. - * - *      2. Redistributions in binary form must reproduce the above - *         copyright notice, this list of conditions and the following - *         disclaimer in the documentation and/or other materials - *         provided with the distribution. - * - * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" - * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE - * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE - * ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE - * LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR - * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF - * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS - * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN - * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) - * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE - * POSSIBILITY OF SUCH DAMAGE. - */ +/* Copyright (C) 2018 Netronome Systems, Inc. */  #define _GNU_SOURCE  #include <stdarg.h> @@ -41,6 +7,7 @@  #include <stdlib.h>  #include <string.h>  #include <sys/types.h> +#include <libbpf.h>  #include "disasm.h"  #include "json_writer.h" @@ -114,7 +81,7 @@ struct kernel_sym *kernel_syms_search(struct dump_data *dd,  		       sizeof(*dd->sym_mapping), kernel_syms_cmp) : NULL;  } -static void print_insn(void *private_data, const char *fmt, ...) +static void __printf(2, 3) print_insn(void *private_data, const char *fmt, ...)  {  	va_list args; @@ -123,7 +90,7 @@ static void print_insn(void *private_data, const char *fmt, ...)  	va_end(args);  } -static void +static void __printf(2, 3)  print_insn_for_graph(void *private_data, const char *fmt, ...)  {  	char buf[64], *p; @@ -154,7 +121,8 @@ print_insn_for_graph(void *private_data, const char *fmt, ...)  	printf("%s", buf);  } -static void print_insn_json(void *private_data, const char *fmt, ...) +static void __printf(2, 3) +print_insn_json(void *private_data, const char *fmt, ...)  {  	unsigned int l = strlen(fmt);  	char chomped_fmt[l]; @@ -234,19 +202,25 @@ static const char *print_imm(void *private_data,  }  void dump_xlated_json(struct dump_data *dd, void *buf, unsigned int len, -		      bool opcodes) +		      bool opcodes, bool linum)  { +	const struct bpf_prog_linfo *prog_linfo = dd->prog_linfo;  	const struct bpf_insn_cbs cbs = {  		.cb_print	= print_insn_json,  		.cb_call	= print_call,  		.cb_imm		= print_imm,  		.private_data	= dd,  	}; +	struct bpf_func_info *record;  	struct bpf_insn *insn = buf; +	struct btf *btf = dd->btf;  	bool double_insn = false; +	unsigned int nr_skip = 0; +	char func_sig[1024];  	unsigned int i;  	jsonw_start_array(json_wtr); +	record = dd->func_info;  	for (i = 0; i < len / sizeof(*insn); i++) {  		if (double_insn) {  			double_insn = false; @@ -255,6 +229,30 @@ void dump_xlated_json(struct dump_data *dd, void *buf, unsigned int len,  		double_insn = insn[i].code == (BPF_LD | BPF_IMM | BPF_DW);  		jsonw_start_object(json_wtr); + +		if (btf && record) { +			if (record->insn_off == i) { +				btf_dumper_type_only(btf, record->type_id, +						     func_sig, +						     sizeof(func_sig)); +				if (func_sig[0] != '\0') { +					jsonw_name(json_wtr, "proto"); +					jsonw_string(json_wtr, func_sig); +				} +				record = (void *)record + dd->finfo_rec_size; +			} +		} + +		if (prog_linfo) { +			const struct bpf_line_info *linfo; + +			linfo = bpf_prog_linfo__lfind(prog_linfo, i, nr_skip); +			if (linfo) { +				btf_dump_linfo_json(btf, linfo, linum); +				nr_skip++; +			} +		} +  		jsonw_name(json_wtr, "disasm");  		print_bpf_insn(&cbs, insn + i, true); @@ -289,24 +287,52 @@ void dump_xlated_json(struct dump_data *dd, void *buf, unsigned int len,  }  void dump_xlated_plain(struct dump_data *dd, void *buf, unsigned int len, -		       bool opcodes) +		       bool opcodes, bool linum)  { +	const struct bpf_prog_linfo *prog_linfo = dd->prog_linfo;  	const struct bpf_insn_cbs cbs = {  		.cb_print	= print_insn,  		.cb_call	= print_call,  		.cb_imm		= print_imm,  		.private_data	= dd,  	}; +	struct bpf_func_info *record;  	struct bpf_insn *insn = buf; +	struct btf *btf = dd->btf; +	unsigned int nr_skip = 0;  	bool double_insn = false; +	char func_sig[1024];  	unsigned int i; +	record = dd->func_info;  	for (i = 0; i < len / sizeof(*insn); i++) {  		if (double_insn) {  			double_insn = false;  			continue;  		} +		if (btf && record) { +			if (record->insn_off == i) { +				btf_dumper_type_only(btf, record->type_id, +						     func_sig, +						     sizeof(func_sig)); +				if (func_sig[0] != '\0') +					printf("%s:\n", func_sig); +				record = (void *)record + dd->finfo_rec_size; +			} +		} + +		if (prog_linfo) { +			const struct bpf_line_info *linfo; + +			linfo = bpf_prog_linfo__lfind(prog_linfo, i, nr_skip); +			if (linfo) { +				btf_dump_linfo_plain(btf, linfo, "; ", +						     linum); +				nr_skip++; +			} +		} +  		double_insn = insn[i].code == (BPF_LD | BPF_IMM | BPF_DW);  		printf("% 4d: ", i); diff --git a/tools/bpf/bpftool/xlated_dumper.h b/tools/bpf/bpftool/xlated_dumper.h index 33d86e2b369b..54847e174273 100644 --- a/tools/bpf/bpftool/xlated_dumper.h +++ b/tools/bpf/bpftool/xlated_dumper.h @@ -1,45 +1,13 @@ -// SPDX-License-Identifier: (GPL-2.0-only OR BSD-2-Clause) -/* - * Copyright (C) 2018 Netronome Systems, Inc. - * - * This software is dual licensed under the GNU General License Version 2, - * June 1991 as shown in the file COPYING in the top-level directory of this - * source tree or the BSD 2-Clause License provided below.  You have the - * option to license this software under the complete terms of either license. - * - * The BSD 2-Clause License: - * - *     Redistribution and use in source and binary forms, with or - *     without modification, are permitted provided that the following - *     conditions are met: - * - *      1. Redistributions of source code must retain the above - *         copyright notice, this list of conditions and the following - *         disclaimer. - * - *      2. Redistributions in binary form must reproduce the above - *         copyright notice, this list of conditions and the following - *         disclaimer in the documentation and/or other materials - *         provided with the distribution. - * - * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" - * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE - * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE - * ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE - * LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR - * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF - * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS - * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN - * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) - * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE - * POSSIBILITY OF SUCH DAMAGE. - */ +/* SPDX-License-Identifier: (GPL-2.0-only OR BSD-2-Clause) */ +/* Copyright (C) 2018 Netronome Systems, Inc. */  #ifndef __BPF_TOOL_XLATED_DUMPER_H  #define __BPF_TOOL_XLATED_DUMPER_H  #define SYM_MAX_NAME	256 +struct bpf_prog_linfo; +  struct kernel_sym {  	unsigned long address;  	char name[SYM_MAX_NAME]; @@ -51,6 +19,10 @@ struct dump_data {  	__u32 sym_count;  	__u64 *jited_ksyms;  	__u32 nr_jited_ksyms; +	struct btf *btf; +	void *func_info; +	__u32 finfo_rec_size; +	const struct bpf_prog_linfo *prog_linfo;  	char scratch_buff[SYM_MAX_NAME + 8];  }; @@ -58,9 +30,9 @@ void kernel_syms_load(struct dump_data *dd);  void kernel_syms_destroy(struct dump_data *dd);  struct kernel_sym *kernel_syms_search(struct dump_data *dd, unsigned long key);  void dump_xlated_json(struct dump_data *dd, void *buf, unsigned int len, -		      bool opcodes); +		       bool opcodes, bool linum);  void dump_xlated_plain(struct dump_data *dd, void *buf, unsigned int len, -		       bool opcodes); +		       bool opcodes, bool linum);  void dump_xlated_for_graph(struct dump_data *dd, void *buf, void *buf_end,  			   unsigned int start_index); | 

