#!/bin/bash
|
# SPDX-License-Identifier: GPL-2.0-only
|
# Generate tags or cscope files
|
# Usage tags.sh <mode>
|
#
|
# mode may be any of: tags, TAGS, cscope
|
#
|
# Uses the following environment variables:
|
# SUBARCH, SRCARCH, srctree
|
|
if [ "$KBUILD_VERBOSE" = "1" ]; then
|
set -x
|
fi
|
|
# RCS_FIND_IGNORE has escaped ()s -- remove them.
|
ignore="$(echo "$RCS_FIND_IGNORE" | sed 's|\\||g' )"
|
# tags and cscope files should also ignore MODVERSION *.mod.c files
|
ignore="$ignore ( -name *.mod.c ) -prune -o"
|
|
# Use make KBUILD_ABS_SRCTREE=1 {tags|cscope}
|
# to force full paths for a non-O= build
|
if [ "${srctree}" = "." -o -z "${srctree}" ]; then
|
tree=
|
else
|
tree=${srctree}/
|
fi
|
|
# ignore userspace tools
|
if [ -n "$COMPILED_SOURCE" ]; then
|
ignore="$ignore ( -path ./tools ) -prune -o"
|
else
|
ignore="$ignore ( -path ${tree}tools ) -prune -o"
|
fi
|
|
# Detect if ALLSOURCE_ARCHS is set. If not, we assume SRCARCH
|
if [ "${ALLSOURCE_ARCHS}" = "" ]; then
|
ALLSOURCE_ARCHS=${SRCARCH}
|
elif [ "${ALLSOURCE_ARCHS}" = "all" ]; then
|
ALLSOURCE_ARCHS=$(find ${tree}arch/ -mindepth 1 -maxdepth 1 -type d -printf '%f ')
|
fi
|
|
# find sources in arch/$1
|
find_arch_sources()
|
{
|
for i in $archincludedir; do
|
prune="$prune -wholename $i -prune -o"
|
done
|
find ${tree}arch/$1 $ignore $prune -name "$2" -not -type l -print;
|
}
|
|
# find sources in arch/$1/include
|
find_arch_include_sources()
|
{
|
include=$(find ${tree}arch/$1/ -name include -type d -print);
|
if [ -n "$include" ]; then
|
archincludedir="$archincludedir $include"
|
find $include $ignore -name "$2" -not -type l -print;
|
fi
|
}
|
|
# find sources in include/
|
find_include_sources()
|
{
|
find ${tree}include $ignore -name config -prune -o -name "$1" \
|
-not -type l -print;
|
}
|
|
# find sources in rest of tree
|
# we could benefit from a list of dirs to search in here
|
find_other_sources()
|
{
|
find ${tree}* $ignore \
|
\( -path ${tree}include -o -path ${tree}arch -o -name '.tmp_*' \) -prune -o \
|
-name "$1" -not -type l -print;
|
}
|
|
find_sources()
|
{
|
find_arch_sources $1 "$2"
|
}
|
|
all_sources()
|
{
|
find_arch_include_sources ${SRCARCH} '*.[chS]'
|
if [ ! -z "$archinclude" ]; then
|
find_arch_include_sources $archinclude '*.[chS]'
|
fi
|
find_include_sources '*.[chS]'
|
for arch in $ALLSOURCE_ARCHS
|
do
|
find_sources $arch '*.[chS]'
|
done
|
find_other_sources '*.[chS]'
|
}
|
|
all_compiled_sources()
|
{
|
realpath -es $([ -z "$KBUILD_ABS_SRCTREE" ] && echo --relative-to=.) \
|
include/generated/autoconf.h $(find $ignore -name "*.cmd" -exec \
|
grep -Poh '(?(?=^source_.* \K).*|(?=^ \K\S).*(?= \\))' {} \+ |
|
awk '!a[$0]++') | sort -u
|
}
|
|
all_target_sources()
|
{
|
if [ -n "$COMPILED_SOURCE" ]; then
|
all_compiled_sources
|
else
|
all_sources
|
fi
|
}
|
|
all_kconfigs()
|
{
|
find ${tree}arch/ -maxdepth 1 $ignore \
|
-name "Kconfig*" -not -type l -print;
|
for arch in $ALLSOURCE_ARCHS; do
|
find_sources $arch 'Kconfig*'
|
done
|
find_other_sources 'Kconfig*'
|
}
|
|
docscope()
|
{
|
(echo \-k; echo \-q; all_target_sources) > cscope.files
|
cscope -b -f cscope.out
|
}
|
|
dogtags()
|
{
|
all_target_sources | gtags -i -f -
|
}
|
|
# Basic regular expressions with an optional /kind-spec/ for ctags and
|
# the following limitations:
|
# - No regex modifiers
|
# - Use \{0,1\} instead of \?, because etags expects an unescaped ?
|
# - \s is not working with etags, use a space or [ \t]
|
# - \w works, but does not match underscores in etags
|
# - etags regular expressions have to match at the start of a line;
|
# a ^[^#] is prepended by setup_regex unless an anchor is already present
|
regex_asm=(
|
'/^\(ENTRY\|_GLOBAL\)(\([[:alnum:]_\\]*\)).*/\2/'
|
)
|
regex_c=(
|
'/^SYSCALL_DEFINE[0-9](\([[:alnum:]_]*\).*/sys_\1/'
|
'/^BPF_CALL_[0-9](\([[:alnum:]_]*\).*/\1/'
|
'/^COMPAT_SYSCALL_DEFINE[0-9](\([[:alnum:]_]*\).*/compat_sys_\1/'
|
'/^TRACE_EVENT(\([[:alnum:]_]*\).*/trace_\1/'
|
'/^TRACE_EVENT(\([[:alnum:]_]*\).*/trace_\1_rcuidle/'
|
'/^DEFINE_EVENT([^,)]*, *\([[:alnum:]_]*\).*/trace_\1/'
|
'/^DEFINE_EVENT([^,)]*, *\([[:alnum:]_]*\).*/trace_\1_rcuidle/'
|
'/^DEFINE_INSN_CACHE_OPS(\([[:alnum:]_]*\).*/get_\1_slot/'
|
'/^DEFINE_INSN_CACHE_OPS(\([[:alnum:]_]*\).*/free_\1_slot/'
|
'/^PAGEFLAG(\([[:alnum:]_]*\).*/Page\1/'
|
'/^PAGEFLAG(\([[:alnum:]_]*\).*/SetPage\1/'
|
'/^PAGEFLAG(\([[:alnum:]_]*\).*/ClearPage\1/'
|
'/^TESTSETFLAG(\([[:alnum:]_]*\).*/TestSetPage\1/'
|
'/^TESTPAGEFLAG(\([[:alnum:]_]*\).*/Page\1/'
|
'/^SETPAGEFLAG(\([[:alnum:]_]*\).*/SetPage\1/'
|
'/\<__SETPAGEFLAG(\([[:alnum:]_]*\).*/__SetPage\1/'
|
'/\<TESTCLEARFLAG(\([[:alnum:]_]*\).*/TestClearPage\1/'
|
'/\<__TESTCLEARFLAG(\([[:alnum:]_]*\).*/TestClearPage\1/'
|
'/\<CLEARPAGEFLAG(\([[:alnum:]_]*\).*/ClearPage\1/'
|
'/\<__CLEARPAGEFLAG(\([[:alnum:]_]*\).*/__ClearPage\1/'
|
'/^__PAGEFLAG(\([[:alnum:]_]*\).*/__SetPage\1/'
|
'/^__PAGEFLAG(\([[:alnum:]_]*\).*/__ClearPage\1/'
|
'/^PAGEFLAG_FALSE(\([[:alnum:]_]*\).*/Page\1/'
|
'/\<TESTSCFLAG(\([[:alnum:]_]*\).*/TestSetPage\1/'
|
'/\<TESTSCFLAG(\([[:alnum:]_]*\).*/TestClearPage\1/'
|
'/\<SETPAGEFLAG_NOOP(\([[:alnum:]_]*\).*/SetPage\1/'
|
'/\<CLEARPAGEFLAG_NOOP(\([[:alnum:]_]*\).*/ClearPage\1/'
|
'/\<__CLEARPAGEFLAG_NOOP(\([[:alnum:]_]*\).*/__ClearPage\1/'
|
'/\<TESTCLEARFLAG_FALSE(\([[:alnum:]_]*\).*/TestClearPage\1/'
|
'/^PAGE_TYPE_OPS(\([[:alnum:]_]*\).*/Page\1/'
|
'/^PAGE_TYPE_OPS(\([[:alnum:]_]*\).*/__SetPage\1/'
|
'/^PAGE_TYPE_OPS(\([[:alnum:]_]*\).*/__ClearPage\1/'
|
'/^TASK_PFA_TEST([^,]*, *\([[:alnum:]_]*\))/task_\1/'
|
'/^TASK_PFA_SET([^,]*, *\([[:alnum:]_]*\))/task_set_\1/'
|
'/^TASK_PFA_CLEAR([^,]*, *\([[:alnum:]_]*\))/task_clear_\1/'
|
'/^DEF_MMIO_\(IN\|OUT\)_[XD](\([[:alnum:]_]*\),[^)]*)/\2/'
|
'/^DEBUGGER_BOILERPLATE(\([[:alnum:]_]*\))/\1/'
|
'/^DEF_PCI_AC_\(\|NO\)RET(\([[:alnum:]_]*\).*/\2/'
|
'/^PCI_OP_READ(\(\w*\).*[1-4])/pci_bus_read_config_\1/'
|
'/^PCI_OP_WRITE(\(\w*\).*[1-4])/pci_bus_write_config_\1/'
|
'/\<DEFINE_\(RT_MUTEX\|MUTEX\|SEMAPHORE\|SPINLOCK\)(\([[:alnum:]_]*\)/\2/v/'
|
'/\<DEFINE_\(RAW_SPINLOCK\|RWLOCK\|SEQLOCK\)(\([[:alnum:]_]*\)/\2/v/'
|
'/\<DECLARE_\(RWSEM\|COMPLETION\)(\([[:alnum:]_]\+\)/\2/v/'
|
'/\<DECLARE_BITMAP(\([[:alnum:]_]*\)/\1/v/'
|
'/\(^\|\s\)\(\|L\|H\)LIST_HEAD(\([[:alnum:]_]*\)/\3/v/'
|
'/\(^\|\s\)RADIX_TREE(\([[:alnum:]_]*\)/\2/v/'
|
'/\<DEFINE_PER_CPU([^,]*, *\([[:alnum:]_]*\)/\1/v/'
|
'/\<DEFINE_PER_CPU_SHARED_ALIGNED([^,]*, *\([[:alnum:]_]*\)/\1/v/'
|
'/\<DECLARE_WAIT_QUEUE_HEAD(\([[:alnum:]_]*\)/\1/v/'
|
'/\<DECLARE_\(TASKLET\|WORK\|DELAYED_WORK\)(\([[:alnum:]_]*\)/\2/v/'
|
'/\(^\s\)OFFSET(\([[:alnum:]_]*\)/\2/v/'
|
'/\(^\s\)DEFINE(\([[:alnum:]_]*\)/\2/v/'
|
'/\<\(DEFINE\|DECLARE\)_HASHTABLE(\([[:alnum:]_]*\)/\2/v/'
|
'/\<DEFINE_ID\(R\|A\)(\([[:alnum:]_]\+\)/\2/'
|
'/\<DEFINE_WD_CLASS(\([[:alnum:]_]\+\)/\1/'
|
'/\<ATOMIC_NOTIFIER_HEAD(\([[:alnum:]_]\+\)/\1/'
|
'/\<RAW_NOTIFIER_HEAD(\([[:alnum:]_]\+\)/\1/'
|
'/\<DECLARE_FAULT_ATTR(\([[:alnum:]_]\+\)/\1/'
|
'/\<BLOCKING_NOTIFIER_HEAD(\([[:alnum:]_]\+\)/\1/'
|
'/\<DEVICE_ATTR_\(RW\|RO\|WO\)(\([[:alnum:]_]\+\)/dev_attr_\2/'
|
'/\<DRIVER_ATTR_\(RW\|RO\|WO\)(\([[:alnum:]_]\+\)/driver_attr_\2/'
|
'/\<\(DEFINE\|DECLARE\)_STATIC_KEY_\(TRUE\|FALSE\)\(\|_RO\)(\([[:alnum:]_]\+\)/\4/'
|
'/^SEQCOUNT_LOCKTYPE(\([^,]*\),[[:space:]]*\([^,]*\),[^)]*)/seqcount_\2_t/'
|
'/^SEQCOUNT_LOCKTYPE(\([^,]*\),[[:space:]]*\([^,]*\),[^)]*)/seqcount_\2_init/'
|
)
|
regex_kconfig=(
|
'/^[[:blank:]]*\(menu\|\)config[[:blank:]]\+\([[:alnum:]_]\+\)/\2/'
|
'/^[[:blank:]]*\(menu\|\)config[[:blank:]]\+\([[:alnum:]_]\+\)/CONFIG_\2/'
|
)
|
setup_regex()
|
{
|
local mode=$1 lang tmp=() r
|
shift
|
|
regex=()
|
for lang; do
|
case "$lang" in
|
asm) tmp=("${regex_asm[@]}") ;;
|
c) tmp=("${regex_c[@]}") ;;
|
kconfig) tmp=("${regex_kconfig[@]}") ;;
|
esac
|
for r in "${tmp[@]}"; do
|
if test "$mode" = "exuberant"; then
|
regex[${#regex[@]}]="--regex-$lang=${r}b"
|
else
|
# Remove ctags /kind-spec/
|
case "$r" in
|
/*/*/?/)
|
r=${r%?/}
|
esac
|
# Prepend ^[^#] unless already anchored
|
case "$r" in
|
/^*) ;;
|
*)
|
r="/^[^#]*${r#/}"
|
esac
|
regex[${#regex[@]}]="--regex=$r"
|
fi
|
done
|
done
|
}
|
|
exuberant()
|
{
|
setup_regex exuberant asm c
|
all_target_sources | xargs $1 -a \
|
-I __initdata,__exitdata,__initconst,__ro_after_init \
|
-I __initdata_memblock \
|
-I __refdata,__attribute,__maybe_unused,__always_unused \
|
-I __acquires,__releases,__deprecated,__always_inline \
|
-I __read_mostly,__aligned,____cacheline_aligned \
|
-I ____cacheline_aligned_in_smp \
|
-I __cacheline_aligned,__cacheline_aligned_in_smp \
|
-I ____cacheline_internodealigned_in_smp \
|
-I __used,__packed,__packed2__,__must_check,__must_hold \
|
-I EXPORT_SYMBOL,EXPORT_SYMBOL_GPL,ACPI_EXPORT_SYMBOL \
|
-I DEFINE_TRACE,EXPORT_TRACEPOINT_SYMBOL,EXPORT_TRACEPOINT_SYMBOL_GPL \
|
-I static,const \
|
--extra=+fq --c-kinds=+px --fields=+iaS --langmap=c:+.h \
|
"${regex[@]}"
|
|
setup_regex exuberant kconfig
|
all_kconfigs | xargs $1 -a \
|
--langdef=kconfig --language-force=kconfig "${regex[@]}"
|
|
}
|
|
emacs()
|
{
|
setup_regex emacs asm c
|
all_target_sources | xargs $1 -a "${regex[@]}"
|
|
setup_regex emacs kconfig
|
all_kconfigs | xargs $1 -a "${regex[@]}"
|
}
|
|
xtags()
|
{
|
if $1 --version 2>&1 | grep -iq exuberant; then
|
exuberant $1
|
elif $1 --version 2>&1 | grep -iq emacs; then
|
emacs $1
|
else
|
all_target_sources | xargs $1 -a
|
fi
|
}
|
|
# Support um (which uses SUBARCH)
|
if [ "${ARCH}" = "um" ]; then
|
if [ "$SUBARCH" = "i386" ]; then
|
archinclude=x86
|
elif [ "$SUBARCH" = "x86_64" ]; then
|
archinclude=x86
|
else
|
archinclude=${SUBARCH}
|
fi
|
fi
|
|
remove_structs=
|
case "$1" in
|
"cscope")
|
docscope
|
;;
|
|
"gtags")
|
dogtags
|
;;
|
|
"tags")
|
rm -f tags
|
xtags ctags
|
remove_structs=y
|
;;
|
|
"TAGS")
|
rm -f TAGS
|
xtags etags
|
remove_structs=y
|
;;
|
esac
|
|
# Remove structure forward declarations.
|
if [ -n "$remove_structs" ]; then
|
LANG=C sed -i -e '/^\([a-zA-Z_][a-zA-Z0-9_]*\)\t.*\t\/\^struct \1;.*\$\/;"\tx$/d' $1
|
fi
|