1# SPDX-License-Identifier: (GPL-2.0-only OR BSD-2-Clause) 2include ../../scripts/Makefile.include 3 4ifeq ($(srctree),) 5srctree := $(patsubst %/,%,$(dir $(CURDIR))) 6srctree := $(patsubst %/,%,$(dir $(srctree))) 7srctree := $(patsubst %/,%,$(dir $(srctree))) 8endif 9 10ifeq ($(V),1) 11 Q = 12else 13 Q = @ 14endif 15 16BPF_DIR = $(srctree)/tools/lib/bpf 17 18ifneq ($(OUTPUT),) 19 _OUTPUT := $(OUTPUT) 20else 21 _OUTPUT := $(CURDIR)/ 22endif 23BOOTSTRAP_OUTPUT := $(_OUTPUT)bootstrap/ 24 25LIBBPF_OUTPUT := $(_OUTPUT)libbpf/ 26LIBBPF_DESTDIR := $(LIBBPF_OUTPUT) 27LIBBPF_INCLUDE := $(LIBBPF_DESTDIR)include 28LIBBPF_HDRS_DIR := $(LIBBPF_INCLUDE)/bpf 29LIBBPF := $(LIBBPF_OUTPUT)libbpf.a 30 31LIBBPF_BOOTSTRAP_OUTPUT := $(BOOTSTRAP_OUTPUT)libbpf/ 32LIBBPF_BOOTSTRAP_DESTDIR := $(LIBBPF_BOOTSTRAP_OUTPUT) 33LIBBPF_BOOTSTRAP_INCLUDE := $(LIBBPF_BOOTSTRAP_DESTDIR)include 34LIBBPF_BOOTSTRAP_HDRS_DIR := $(LIBBPF_BOOTSTRAP_INCLUDE)/bpf 35LIBBPF_BOOTSTRAP := $(LIBBPF_BOOTSTRAP_OUTPUT)libbpf.a 36 37# We need to copy hashmap.h, nlattr.h, relo_core.h and libbpf_internal.h 38# which are not otherwise exported by libbpf, but still required by bpftool. 39LIBBPF_INTERNAL_HDRS := $(addprefix $(LIBBPF_HDRS_DIR)/,hashmap.h nlattr.h relo_core.h libbpf_internal.h) 40LIBBPF_BOOTSTRAP_INTERNAL_HDRS := $(addprefix $(LIBBPF_BOOTSTRAP_HDRS_DIR)/,hashmap.h relo_core.h libbpf_internal.h) 41 42$(LIBBPF_OUTPUT) $(BOOTSTRAP_OUTPUT) $(LIBBPF_BOOTSTRAP_OUTPUT) $(LIBBPF_HDRS_DIR) $(LIBBPF_BOOTSTRAP_HDRS_DIR): 43 $(QUIET_MKDIR)mkdir -p $@ 44 45$(LIBBPF): $(wildcard $(BPF_DIR)/*.[ch] $(BPF_DIR)/Makefile) | $(LIBBPF_OUTPUT) 46 $(Q)$(MAKE) -C $(BPF_DIR) OUTPUT=$(LIBBPF_OUTPUT) \ 47 DESTDIR=$(LIBBPF_DESTDIR:/=) prefix= $(LIBBPF) install_headers 48 49$(LIBBPF_INTERNAL_HDRS): $(LIBBPF_HDRS_DIR)/%.h: $(BPF_DIR)/%.h | $(LIBBPF_HDRS_DIR) 50 $(call QUIET_INSTALL, $@) 51 $(Q)install -m 644 -t $(LIBBPF_HDRS_DIR) $< 52 53$(LIBBPF_BOOTSTRAP): $(wildcard $(BPF_DIR)/*.[ch] $(BPF_DIR)/Makefile) | $(LIBBPF_BOOTSTRAP_OUTPUT) 54 $(Q)$(MAKE) -C $(BPF_DIR) OUTPUT=$(LIBBPF_BOOTSTRAP_OUTPUT) \ 55 DESTDIR=$(LIBBPF_BOOTSTRAP_DESTDIR:/=) prefix= \ 56 ARCH= CROSS_COMPILE= CC="$(HOSTCC)" LD="$(HOSTLD)" AR="$(HOSTAR)" $@ install_headers 57 58$(LIBBPF_BOOTSTRAP_INTERNAL_HDRS): $(LIBBPF_BOOTSTRAP_HDRS_DIR)/%.h: $(BPF_DIR)/%.h | $(LIBBPF_BOOTSTRAP_HDRS_DIR) 59 $(call QUIET_INSTALL, $@) 60 $(Q)install -m 644 -t $(LIBBPF_BOOTSTRAP_HDRS_DIR) $< 61 62$(LIBBPF)-clean: FORCE | $(LIBBPF_OUTPUT) 63 $(call QUIET_CLEAN, libbpf) 64 $(Q)$(MAKE) -C $(BPF_DIR) OUTPUT=$(LIBBPF_OUTPUT) clean >/dev/null 65 66$(LIBBPF_BOOTSTRAP)-clean: FORCE | $(LIBBPF_BOOTSTRAP_OUTPUT) 67 $(call QUIET_CLEAN, libbpf-bootstrap) 68 $(Q)$(MAKE) -C $(BPF_DIR) OUTPUT=$(LIBBPF_BOOTSTRAP_OUTPUT) clean >/dev/null 69 70prefix ?= /usr/local 71bash_compdir ?= /usr/share/bash-completion/completions 72 73CFLAGS += -O2 74CFLAGS += -W -Wall -Wextra -Wno-unused-parameter -Wno-missing-field-initializers 75CFLAGS += $(filter-out -Wswitch-enum -Wnested-externs,$(EXTRA_WARNINGS)) 76CFLAGS += -DPACKAGE='"bpftool"' -D__EXPORTED_HEADERS__ \ 77 -I$(or $(OUTPUT),.) \ 78 -I$(LIBBPF_INCLUDE) \ 79 -I$(srctree)/kernel/bpf/ \ 80 -I$(srctree)/tools/include \ 81 -I$(srctree)/tools/include/uapi 82ifneq ($(BPFTOOL_VERSION),) 83CFLAGS += -DBPFTOOL_VERSION='"$(BPFTOOL_VERSION)"' 84endif 85ifneq ($(EXTRA_CFLAGS),) 86CFLAGS += $(EXTRA_CFLAGS) 87endif 88ifneq ($(EXTRA_LDFLAGS),) 89LDFLAGS += $(EXTRA_LDFLAGS) 90endif 91 92HOST_CFLAGS := $(subst -I$(LIBBPF_INCLUDE),-I$(LIBBPF_BOOTSTRAP_INCLUDE),\ 93 $(subst $(CLANG_CROSS_FLAGS),,$(CFLAGS))) 94HOST_LDFLAGS := $(LDFLAGS) 95 96INSTALL ?= install 97RM ?= rm -f 98 99FEATURE_USER = .bpftool 100 101FEATURE_TESTS := clang-bpf-co-re 102FEATURE_TESTS += llvm 103FEATURE_TESTS += libcap 104FEATURE_TESTS += libbfd 105FEATURE_TESTS += libbfd-liberty 106FEATURE_TESTS += libbfd-liberty-z 107FEATURE_TESTS += disassembler-four-args 108FEATURE_TESTS += disassembler-init-styled 109FEATURE_TESTS += libelf-zstd 110 111FEATURE_DISPLAY := clang-bpf-co-re 112FEATURE_DISPLAY += llvm 113FEATURE_DISPLAY += libcap 114FEATURE_DISPLAY += libbfd 115FEATURE_DISPLAY += libbfd-liberty 116FEATURE_DISPLAY += libbfd-liberty-z 117 118check_feat := 1 119NON_CHECK_FEAT_TARGETS := clean uninstall doc doc-clean doc-install doc-uninstall 120ifdef MAKECMDGOALS 121ifeq ($(filter-out $(NON_CHECK_FEAT_TARGETS),$(MAKECMDGOALS)),) 122 check_feat := 0 123endif 124endif 125 126ifeq ($(check_feat),1) 127ifeq ($(FEATURES_DUMP),) 128include $(srctree)/tools/build/Makefile.feature 129else 130include $(FEATURES_DUMP) 131endif 132endif 133 134LIBS = $(LIBBPF) -lelf -lz 135LIBS_BOOTSTRAP = $(LIBBPF_BOOTSTRAP) -lelf -lz 136 137ifeq ($(feature-libelf-zstd),1) 138LIBS += -lzstd 139LIBS_BOOTSTRAP += -lzstd 140endif 141 142ifeq ($(feature-libcap), 1) 143CFLAGS += -DUSE_LIBCAP 144LIBS += -lcap 145endif 146 147include $(wildcard $(OUTPUT)*.d) 148 149all: $(OUTPUT)bpftool 150 151SRCS := $(wildcard *.c) 152 153ifeq ($(feature-llvm),1) 154 # If LLVM is available, use it for JIT disassembly 155 CFLAGS += -DHAVE_LLVM_SUPPORT 156 LLVM_CONFIG_LIB_COMPONENTS := mcdisassembler all-targets 157 # llvm-config always adds -D_GNU_SOURCE, however, it may already be in CFLAGS 158 # (e.g. when bpftool build is called from selftests build as selftests 159 # Makefile includes lib.mk which sets -D_GNU_SOURCE) which would cause 160 # compilation error due to redefinition. Let's filter it out here. 161 CFLAGS += $(filter-out -D_GNU_SOURCE,$(shell $(LLVM_CONFIG) --cflags)) 162 LIBS += $(shell $(LLVM_CONFIG) --libs $(LLVM_CONFIG_LIB_COMPONENTS)) 163 ifeq ($(shell $(LLVM_CONFIG) --shared-mode),static) 164 LIBS += $(shell $(LLVM_CONFIG) --system-libs $(LLVM_CONFIG_LIB_COMPONENTS)) 165 LIBS += -lstdc++ 166 endif 167 LDFLAGS += $(shell $(LLVM_CONFIG) --ldflags) 168else 169 # Fall back on libbfd 170 ifeq ($(feature-libbfd),1) 171 LIBS += -lbfd -ldl -lopcodes 172 else ifeq ($(feature-libbfd-liberty),1) 173 LIBS += -lbfd -ldl -lopcodes -liberty 174 else ifeq ($(feature-libbfd-liberty-z),1) 175 LIBS += -lbfd -ldl -lopcodes -liberty -lz 176 endif 177 178 # If one of the above feature combinations is set, we support libbfd 179 ifneq ($(filter -lbfd,$(LIBS)),) 180 CFLAGS += -DHAVE_LIBBFD_SUPPORT 181 182 # Libbfd interface changed over time, figure out what we need 183 ifeq ($(feature-disassembler-four-args), 1) 184 CFLAGS += -DDISASM_FOUR_ARGS_SIGNATURE 185 endif 186 ifeq ($(feature-disassembler-init-styled), 1) 187 CFLAGS += -DDISASM_INIT_STYLED 188 endif 189 endif 190endif 191ifeq ($(filter -DHAVE_LLVM_SUPPORT -DHAVE_LIBBFD_SUPPORT,$(CFLAGS)),) 192 # No support for JIT disassembly 193 SRCS := $(filter-out jit_disasm.c,$(SRCS)) 194endif 195 196BPFTOOL_BOOTSTRAP := $(BOOTSTRAP_OUTPUT)bpftool 197 198BOOTSTRAP_OBJS = $(addprefix $(BOOTSTRAP_OUTPUT),main.o common.o json_writer.o gen.o btf.o) 199$(BOOTSTRAP_OBJS): $(LIBBPF_BOOTSTRAP) 200 201OBJS = $(patsubst %.c,$(OUTPUT)%.o,$(SRCS)) $(OUTPUT)disasm.o 202$(OBJS): $(LIBBPF) $(LIBBPF_INTERNAL_HDRS) 203 204VMLINUX_BTF_PATHS ?= $(if $(O),$(O)/vmlinux) \ 205 $(if $(KBUILD_OUTPUT),$(KBUILD_OUTPUT)/vmlinux) \ 206 ../../../vmlinux \ 207 /sys/kernel/btf/vmlinux \ 208 /boot/vmlinux-$(shell uname -r) 209VMLINUX_BTF ?= $(abspath $(firstword $(wildcard $(VMLINUX_BTF_PATHS)))) 210 211bootstrap: $(BPFTOOL_BOOTSTRAP) 212 213ifneq ($(VMLINUX_BTF)$(VMLINUX_H),) 214ifeq ($(feature-clang-bpf-co-re),1) 215 216BUILD_BPF_SKELS := 1 217 218ifeq ($(VMLINUX_H),) 219$(OUTPUT)vmlinux.h: $(VMLINUX_BTF) $(BPFTOOL_BOOTSTRAP) 220 $(QUIET_GEN)$(BPFTOOL_BOOTSTRAP) btf dump file $< format c > $@ 221else 222$(OUTPUT)vmlinux.h: $(VMLINUX_H) 223 $(Q)cp "$(VMLINUX_H)" $@ 224endif 225 226$(OUTPUT)%.bpf.o: skeleton/%.bpf.c $(OUTPUT)vmlinux.h $(LIBBPF_BOOTSTRAP) 227 $(QUIET_CLANG)$(CLANG) \ 228 -I$(or $(OUTPUT),.) \ 229 -I$(srctree)/tools/include/uapi/ \ 230 -I$(LIBBPF_BOOTSTRAP_INCLUDE) \ 231 -g -O2 -Wall -fno-stack-protector \ 232 --target=bpf -c $< -o $@ 233 $(Q)$(LLVM_STRIP) -g $@ 234 235$(OUTPUT)%.skel.h: $(OUTPUT)%.bpf.o $(BPFTOOL_BOOTSTRAP) 236 $(QUIET_GEN)$(BPFTOOL_BOOTSTRAP) gen skeleton $< > $@ 237 238$(OUTPUT)prog.o: $(OUTPUT)profiler.skel.h 239 240$(OUTPUT)pids.o: $(OUTPUT)pid_iter.skel.h 241 242endif 243endif 244 245CFLAGS += $(if $(BUILD_BPF_SKELS),,-DBPFTOOL_WITHOUT_SKELETONS) 246 247$(OUTPUT)disasm.o: $(srctree)/kernel/bpf/disasm.c 248 $(QUIET_CC)$(CC) $(CFLAGS) -c -MMD $< -o $@ 249 250$(BPFTOOL_BOOTSTRAP): $(BOOTSTRAP_OBJS) $(LIBBPF_BOOTSTRAP) 251 $(QUIET_LINK)$(HOSTCC) $(HOST_CFLAGS) $(HOST_LDFLAGS) $(BOOTSTRAP_OBJS) $(LIBS_BOOTSTRAP) -o $@ 252 253$(OUTPUT)bpftool: $(OBJS) $(LIBBPF) 254 $(QUIET_LINK)$(CC) $(CFLAGS) $(LDFLAGS) $(OBJS) $(LIBS) -o $@ 255 256$(BOOTSTRAP_OUTPUT)%.o: %.c $(LIBBPF_BOOTSTRAP_INTERNAL_HDRS) | $(BOOTSTRAP_OUTPUT) 257 $(QUIET_CC)$(HOSTCC) $(HOST_CFLAGS) -c -MMD $< -o $@ 258 259$(OUTPUT)%.o: %.c 260 $(QUIET_CC)$(CC) $(CFLAGS) -c -MMD $< -o $@ 261 262feature-detect-clean: 263 $(call QUIET_CLEAN, feature-detect) 264 $(Q)$(MAKE) -C $(srctree)/tools/build/feature/ clean >/dev/null 265 266clean: $(LIBBPF)-clean $(LIBBPF_BOOTSTRAP)-clean feature-detect-clean 267 $(call QUIET_CLEAN, bpftool) 268 $(Q)$(RM) -- $(OUTPUT)bpftool $(OUTPUT)*.o $(OUTPUT)*.d 269 $(Q)$(RM) -- $(OUTPUT)*.skel.h $(OUTPUT)vmlinux.h 270 $(Q)$(RM) -r -- $(LIBBPF_OUTPUT) $(BOOTSTRAP_OUTPUT) 271 $(call QUIET_CLEAN, core-gen) 272 $(Q)$(RM) -- $(OUTPUT)FEATURE-DUMP.bpftool 273 $(Q)$(RM) -r -- $(OUTPUT)feature/ 274 275install-bin: $(OUTPUT)bpftool 276 $(call QUIET_INSTALL, bpftool) 277 $(Q)$(INSTALL) -m 0755 -d $(DESTDIR)$(prefix)/sbin 278 $(Q)$(INSTALL) $(OUTPUT)bpftool $(DESTDIR)$(prefix)/sbin/bpftool 279 280install: install-bin 281 $(Q)$(INSTALL) -m 0755 -d $(DESTDIR)$(bash_compdir) 282 $(Q)$(INSTALL) -m 0644 bash-completion/bpftool $(DESTDIR)$(bash_compdir) 283 284uninstall: 285 $(call QUIET_UNINST, bpftool) 286 $(Q)$(RM) -- $(DESTDIR)$(prefix)/sbin/bpftool 287 $(Q)$(RM) -- $(DESTDIR)$(bash_compdir)/bpftool 288 289doc: 290 $(call descend,Documentation) 291 292doc-clean: 293 $(call descend,Documentation,clean) 294 295doc-install: 296 $(call descend,Documentation,install) 297 298doc-uninstall: 299 $(call descend,Documentation,uninstall) 300 301FORCE: 302 303.SECONDARY: 304.PHONY: all FORCE bootstrap clean install-bin install uninstall 305.PHONY: doc doc-clean doc-install doc-uninstall 306.DEFAULT_GOAL := all 307 308# Delete partially updated (corrupted) files on error 309.DELETE_ON_ERROR: 310