// SPDX-License-Identifier: GPL-2.0-only #include #include #include #include "bpf_util.h" #include "bpftool_helpers.h" #define BPFTOOL_PATH_MAX_LEN 64 #define BPFTOOL_FULL_CMD_MAX_LEN 512 #define BPFTOOL_DEFAULT_PATH "tools/sbin/bpftool" static int detect_bpftool_path(char *buffer, size_t size) { char tmp[BPFTOOL_PATH_MAX_LEN]; const char *env_path; /* First, check if BPFTOOL environment variable is set */ env_path = getenv("BPFTOOL"); if (env_path && access(env_path, X_OK) == 0) { strscpy(buffer, env_path, size); return 0; } else if (env_path) { fprintf(stderr, "bpftool '%s' doesn't exist or is not executable\n", env_path); return 1; } /* Check default bpftool location (will work if we are running the * default flavor of test_progs) */ snprintf(tmp, BPFTOOL_PATH_MAX_LEN, "./%s", BPFTOOL_DEFAULT_PATH); if (access(tmp, X_OK) == 0) { strscpy(buffer, tmp, size); return 0; } /* Check alternate bpftool location (will work if we are running a * specific flavor of test_progs, e.g. cpuv4 or no_alu32) */ snprintf(tmp, BPFTOOL_PATH_MAX_LEN, "../%s", BPFTOOL_DEFAULT_PATH); if (access(tmp, X_OK) == 0) { strscpy(buffer, tmp, size); return 0; } fprintf(stderr, "Failed to detect bpftool path, use BPFTOOL env var to override\n"); return 1; } static int run_command(char *args, char *output_buf, size_t output_max_len) { static char bpftool_path[BPFTOOL_PATH_MAX_LEN] = {0}; bool suppress_output = !(output_buf && output_max_len); char command[BPFTOOL_FULL_CMD_MAX_LEN]; FILE *f; int ret; /* Detect and cache bpftool binary location */ if (bpftool_path[0] == 0 && detect_bpftool_path(bpftool_path, sizeof(bpftool_path))) return 1; ret = snprintf(command, BPFTOOL_FULL_CMD_MAX_LEN, "%s %s%s", bpftool_path, args, suppress_output ? " > /dev/null 2>&1" : ""); f = popen(command, "r"); if (!f) return 1; if (!suppress_output) fread(output_buf, 1, output_max_len, f); ret = pclose(f); return ret; } int run_bpftool_command(char *args) { return run_command(args, NULL, 0); } int get_bpftool_command_output(char *args, char *output_buf, size_t output_max_len) { return run_command(args, output_buf, output_max_len); }