| # SPDX-License-Identifier: GPL-2.0-only |
| # bash completion for GNU make with kbuild extension -*- shell-script -*- |
| |
| # Load the default completion script for make. It is typically located at |
| # /usr/share/bash-completion/completions/make, but we do not rely on it. |
| __kbuild_load_default_make_completion() |
| { |
| local -a dirs=("${BASH_COMPLETION_USER_DIR:-${XDG_DATA_HOME:-$HOME/.local/share}/bash-completion}/completions") |
| local ifs=$IFS IFS=: dir compfile this_dir |
| |
| for dir in ${XDG_DATA_DIRS:-/usr/local/share:/usr/share}; do |
| dirs+=("$dir"/bash-completion/completions) |
| done |
| IFS=$ifs |
| |
| this_dir="$(realpath "$(dirname "${BASH_SOURCE[0]}")")" |
| |
| for dir in "${dirs[@]}"; do |
| if [[ ! -d ${dir} || ${dir} = "${this_dir}" ]]; then |
| continue |
| fi |
| |
| for compfile in make make.bash _make; do |
| compfile=$dir/$compfile |
| # Avoid trying to source dirs; https://bugzilla.redhat.com/903540 |
| if [[ -f ${compfile} ]] && . "${compfile}" &>/dev/null; then |
| |
| __kbuild_default_make_completion=$( |
| # shellcheck disable=SC2046 # word splitting is the point here |
| set -- $(complete -p make) |
| |
| while [[ $# -gt 1 && "$1" != -F ]]; do |
| shift |
| done |
| |
| if [[ "$1" = -F ]]; then |
| echo "$2" |
| fi |
| ) |
| |
| return |
| fi |
| done |
| done |
| } |
| |
| __kbuild_load_default_make_completion |
| |
| __kbuild_handle_variable() |
| { |
| local var=${1%%=*} |
| local cur=${cur#"${var}"=} |
| local srctree=$2 |
| local keywords=() |
| |
| case $var in |
| ARCH) |
| # sub-directories under arch/ |
| keywords+=($(find "${srctree}/arch" -mindepth 1 -maxdepth 1 -type d -printf '%P\n')) |
| # architectures hard-coded in the top Makefile |
| keywords+=(i386 x86_64 sparc32 sparc64 parisc64) |
| ;; |
| CROSS_COMPILE) |
| # toolchains with a full path |
| local cross_compile=() |
| local c c2 |
| _filedir |
| |
| for c in "${COMPREPLY[@]}"; do |
| # eval for tilde expansion |
| # suppress error, as this fails when it contains a space |
| eval "c2=${c}" 2>/dev/null || continue |
| if [[ ${c} == *-elfedit && ! -d ${c2} && -x ${c2} ]]; then |
| cross_compile+=("${c%elfedit}") |
| fi |
| done |
| |
| # toolchains in the PATH environment |
| while read -r c; do |
| if [[ ${c} == *-elfedit ]]; then |
| keywords+=("${c%elfedit}") |
| fi |
| done < <(compgen -c) |
| |
| COMPREPLY=() |
| _filedir -d |
| |
| # Add cross_compile directly without passing it to compgen. |
| # Otherwise, toolchain paths with a tilde do not work. |
| # e.g.) |
| # CROSS_COMPILE=~/0day/gcc-14.2.0-nolibc/aarch64-linux/bin/aarch64-linux- |
| COMPREPLY+=("${cross_compile[@]}") |
| ;; |
| LLVM) |
| # LLVM=1 uses the default 'clang' etc. |
| keywords+=(1) |
| |
| # suffix for a particular version. LLVM=-18 uses 'clang-18' etc. |
| while read -r c; do |
| if [[ ${c} == clang-[0-9]* ]]; then |
| keywords+=("${c#clang}") |
| fi |
| done < <(compgen -c) |
| |
| # directory path to LLVM toolchains |
| _filedir -d |
| ;; |
| KCONFIG_ALLCONFIG) |
| # KCONFIG_ALLCONFIG=1 selects the default fragment |
| keywords+=(1) |
| # or the path to a fragment file |
| _filedir |
| ;; |
| C | KBUILD_CHECKSRC) |
| keywords+=(1 2) |
| ;; |
| V | KBUILD_VERBOSE) |
| keywords+=({,1}{,2}) |
| ;; |
| W | KBUILD_EXTRA_WARN) |
| keywords+=({,1}{,2}{,3}{,c}{,e}) |
| ;; |
| KBUILD_ABS_SRCTREE | KBUILD_MODPOST_NOFINAL | KBUILD_MODPOST_WARN | \ |
| CLIPPY | KBUILD_CLIPPY | KCONFIG_NOSILENTUPDATE | \ |
| KCONFIG_OVERWRITECONFIG | KCONFIG_WARN_UNKNOWN_SYMBOL | \ |
| KCONFIG_WERROR ) |
| keywords+=(1) |
| ;; |
| INSTALL_MOD_STRIP) |
| keywords+=(1 --strip-debug --strip-unneeded) |
| ;; |
| O | KBUILD_OUTPUT | M | KBUILD_EXTMOD | MO | KBUILD_EXTMOD_OUTPUT | *_PATH) |
| # variables that take a directory. |
| _filedir -d |
| return |
| ;; |
| KBUILD_EXTRA_SYMBOL | KBUILD_KCONFIG | KCONFIG_CONFIG) |
| # variables that take a file. |
| _filedir |
| return |
| esac |
| |
| COMPREPLY+=($(compgen -W "${keywords[*]}" -- "${cur}")) |
| } |
| |
| # Check the -C, -f options and 'source' symlink. Return the source tree we are |
| # working in. |
| __kbuild_get_srctree() |
| { |
| local words=("$@") |
| local cwd makef_dir |
| |
| # see if a path was specified with -C/--directory |
| for ((i = 1; i < ${#words[@]}; i++)); do |
| if [[ ${words[i]} == -@(C|-directory) ]]; then |
| # eval for tilde expansion. |
| # suppress error, as this fails when it contains a space |
| eval "cwd=${words[i + 1]}" 2>/dev/null |
| break |
| fi |
| done |
| |
| if [[ -z ${cwd} ]]; then |
| cwd=. |
| fi |
| |
| # see if a Makefile was specified with -f/--file/--makefile |
| for ((i = 1; i < ${#words[@]}; i++)); do |
| if [[ ${words[i]} == -@(f|-?(make)file) ]]; then |
| # eval for tilde expansion |
| # suppress error, as this fails when it contains a space |
| eval "makef_dir=${words[i + 1]%/*}" 2>/dev/null |
| break |
| fi |
| done |
| |
| if [ -z "${makef_dir}" ]; then |
| makef_dir=${cwd} |
| elif [[ ${makef_dir} != /* ]]; then |
| makef_dir=${cwd}/${makef_dir} |
| fi |
| |
| # If ${makef_dir} is a build directory created by the O= option, there |
| # is a symbolic link 'source', which points to the kernel source tree. |
| if [[ -L ${makef_dir}/source ]]; then |
| makef_dir=$(readlink "${makef_dir}/source") |
| fi |
| |
| echo "${makef_dir}" |
| } |
| |
| # Get SRCARCH to do a little more clever things |
| __kbuild_get_srcarch() |
| { |
| local words=("$@") |
| local arch srcarch uname_m |
| |
| # see if ARCH= is explicitly specified |
| for ((i = 1; i < ${#words[@]}; i++)); do |
| if [[ ${words[i]} == ARCH=* ]]; then |
| arch=${words[i]#ARCH=} |
| break |
| fi |
| done |
| |
| # If ARCH= is not specified, check the build marchine's architecture |
| if [[ -z ${arch} ]]; then |
| uname_m=$(uname -m) |
| |
| # shellcheck disable=SC2209 # 'sh' is SuperH, not a shell command |
| case ${uname_m} in |
| arm64 | aarch64*) arch=arm64 ;; |
| arm* | sa110) arch=arm ;; |
| i?86 | x86_64) arch=x86 ;; |
| loongarch*) arch=loongarch ;; |
| mips*) arch=mips ;; |
| ppc*) arch=powerpc ;; |
| riscv*) arch=riscv ;; |
| s390x) arch=s390 ;; |
| sh[234]*) arch=sh ;; |
| sun4u) arch=sparc64 ;; |
| *) arch=${uname_m} ;; |
| esac |
| fi |
| |
| case ${arch} in |
| parisc64) srcarch=parisc ;; |
| sparc32 | sparc64) srcarch=sparc ;; |
| i386 | x86_64) srcarch=x86 ;; |
| *) srcarch=${arch} ;; |
| esac |
| |
| echo "$srcarch" |
| } |
| |
| # small Makefile to parse obj-* syntax |
| __kbuild_tmp_makefile() |
| { |
| cat <<'EOF' |
| .PHONY: __default |
| __default: |
| $(foreach m,$(obj-y) $(obj-m) $(obj-),$(foreach s, -objs -y -m -,$($(m:%.o=%$s))) $(m)) |
| EOF |
| echo "include ${1}" |
| } |
| |
| _make_for_kbuild () |
| { |
| # shellcheck disable=SC2034 # these are set by _init_completion |
| local cur prev words cword split |
| _init_completion -s || return |
| |
| local srctree |
| srctree=$(__kbuild_get_srctree "${words[@]}") |
| |
| # If 'kernel' and 'Documentation' directories are found, we assume this |
| # is a kernel tree. Otherwise, we fall back to the generic rule provided |
| # by the bash-completion project. |
| if [[ ! -d ${srctree}/kernel || ! -d ${srctree}/Documentation ]]; then |
| if [ -n "${__kbuild_default_make_completion}" ]; then |
| "${__kbuild_default_make_completion}" "$@" |
| fi |
| return |
| fi |
| |
| # make options with a parameter (copied from the bash-completion project) |
| case ${prev} in |
| --file | --makefile | --old-file | --assume-old | --what-if | --new-file | \ |
| --assume-new | -!(-*)[foW]) |
| _filedir |
| return |
| ;; |
| --include-dir | --directory | -!(-*)[ICm]) |
| _filedir -d |
| return |
| ;; |
| -!(-*)E) |
| COMPREPLY=($(compgen -v -- "$cur")) |
| return |
| ;; |
| --eval | -!(-*)[DVx]) |
| return |
| ;; |
| --jobs | -!(-*)j) |
| COMPREPLY=($(compgen -W "{1..$(($(_ncpus) * 2))}" -- "$cur")) |
| return |
| ;; |
| esac |
| |
| local keywords=() |
| |
| case ${cur} in |
| -*) |
| # make options (copied from the bash-completion project) |
| local opts |
| opts="$(_parse_help "$1")" |
| COMPREPLY=($(compgen -W "${opts:-$(_parse_usage "$1")}" -- "$cur")) |
| if [[ ${COMPREPLY-} == *= ]]; then |
| compopt -o nospace |
| fi |
| return |
| ;; |
| *=*) |
| __kbuild_handle_variable "${cur}" "${srctree}" |
| return |
| ;; |
| KBUILD_*) |
| # There are many variables prefixed with 'KBUILD_'. |
| # Display them only when 'KBUILD_' is entered. |
| # shellcheck disable=SC2191 # '=' is appended for variables |
| keywords+=( |
| KBUILD_{CHECKSRC,EXTMOD,EXTMOD_OUTPUT,OUTPUT,VERBOSE,EXTRA_WARN,CLIPPY}= |
| KBUILD_BUILD_{USER,HOST,TIMESTAMP}= |
| KBUILD_MODPOST_{NOFINAL,WARN}= |
| KBUILD_{ABS_SRCTREE,EXTRA_SYMBOLS,KCONFIG}= |
| ) |
| ;; |
| KCONFIG_*) |
| # There are many variables prefixed with 'KCONFIG_'. |
| # Display them only when 'KCONFIG_' is entered. |
| # shellcheck disable=SC2191 # '=' is appended for variables |
| keywords+=( |
| KCONFIG_{CONFIG,ALLCONFIG,NOSILENTUPDATE,OVERWRITECONFIG}= |
| KCONFIG_{SEED,PROBABILITY}= |
| KCONFIG_WARN_UNKNOWN_SYMBOL= |
| KCONFIG_WERROR= |
| ) |
| ;; |
| *) |
| # By default, hide KBUILD_* and KCONFIG_* variables. |
| # Instead, display only the prefix parts. |
| keywords+=(KBUILD_ KCONFIG_) |
| ;; |
| esac |
| |
| if [[ ${cur} != /* && ${cur} != *//* ]]; then |
| local dir srcarch kbuild_file tmp |
| srcarch=$(__kbuild_get_srcarch "${words[@]}") |
| |
| # single build |
| dir=${cur} |
| while true; do |
| if [[ ${dir} == */* ]]; then |
| dir=${dir%/*} |
| else |
| dir=. |
| fi |
| |
| # Search for 'Kbuild' or 'Makefile' in the parent |
| # directories (may not be a direct parent) |
| if [[ -f ${srctree}/${dir}/Kbuild ]]; then |
| kbuild_file=${srctree}/${dir}/Kbuild |
| break |
| fi |
| if [[ -f ${srctree}/${dir}/Makefile ]]; then |
| kbuild_file=${srctree}/${dir}/Makefile |
| break |
| fi |
| |
| if [[ ${dir} == . ]]; then |
| break |
| fi |
| done |
| |
| if [[ -n ${kbuild_file} ]]; then |
| tmp=($(__kbuild_tmp_makefile "${kbuild_file}" | |
| SRCARCH=${srcarch} obj=${dir} src=${srctree}/${dir} \ |
| "${1}" -n -f - 2>/dev/null)) |
| |
| # Add $(obj)/ prefix |
| if [[ ${dir} != . ]]; then |
| tmp=("${tmp[@]/#/${dir}\/}") |
| fi |
| |
| keywords+=("${tmp[@]}") |
| fi |
| |
| # *_defconfig and *.config files. These might be grouped into |
| # subdirectories, e.g., arch/powerpc/configs/*/*_defconfig. |
| if [[ ${cur} == */* ]]; then |
| dir=${cur%/*} |
| else |
| dir=. |
| fi |
| |
| tmp=($(find "${srctree}/arch/${srcarch}/configs/${dir}" \ |
| "${srctree}/kernel/configs/${dir}" \ |
| -mindepth 1 -maxdepth 1 -type d -printf '%P/\n' \ |
| -o -printf '%P\n' 2>/dev/null)) |
| |
| if [[ ${dir} != . ]]; then |
| tmp=("${tmp[@]/#/${dir}\/}") |
| fi |
| |
| keywords+=("${tmp[@]}") |
| fi |
| |
| # shellcheck disable=SC2191 # '=' is appended for variables |
| keywords+=( |
| # |
| # variables (append =) |
| # |
| ARCH= |
| CROSS_COMPILE= |
| LLVM= |
| C= M= MO= O= V= W= |
| INSTALL{,_MOD,_HDR,_DTBS}_PATH= |
| KERNELRELEASE= |
| |
| # |
| # targets |
| # |
| all help |
| clean mrproper distclean |
| clang-{tidy,analyzer} compile_commands.json |
| coccicheck |
| dtbs{,_check,_install} dt_binding_{check,schemas} |
| headers{,_install} |
| vmlinux install |
| modules{,_prepare,_install,_sign} |
| vdso_install |
| tags TAGS cscope gtags |
| rust{available,fmt,fmtcheck} |
| kernel{version,release} image_name |
| kselftest{,-all,-install,-clean,-merge} |
| |
| # configuration |
| {,old,olddef,sync,def,savedef,rand,listnew,helpnew,test,tiny}config |
| {,build_}{menu,n,g,x}config |
| local{mod,yes}config |
| all{no,yes,mod,def}config |
| {yes2mod,mod2yes,mod2no}config |
| |
| # docs |
| {html,textinfo,info,latex,pdf,epub,xml,linkcheck,refcheck,clean}docs |
| |
| # package |
| {,bin,src}{rpm,deb}-pkg |
| {pacman,dir,tar}-pkg |
| tar{,gz,bz2,xz,zst}-pkg |
| perf-tar{,gz,bz2,xz,zst}-src-pkg |
| ) |
| |
| COMPREPLY=($(compgen -W "${keywords[*]}" -- "${cur}")) |
| |
| # Do not append a space for variables, subdirs, "KBUILD_", "KCONFIG_". |
| if [[ ${COMPREPLY-} == *[=/] || ${COMPREPLY-} =~ ^(KBUILD|KCONFIG)_$ ]]; then |
| compopt -o nospace |
| fi |
| |
| } && complete -F _make_for_kbuild make |