Linux kernel mirror (for testing) git.kernel.org/pub/scm/linux/kernel/git/torvalds/linux.git
kernel os linux
1
fork

Configure Feed

Select the types of activity you want to include in your feed.

kheaders: rebuild kheaders_data.tar.xz when a file is modified within a minute

When a header file is changed, kernel/gen_kheaders.sh may fail to update
kernel/kheaders_data.tar.xz.

[steps to reproduce]

[1] Build kernel/kheaders_data.tar.xz

$ make -j$(nproc) kernel/kheaders.o
DESCEND objtool
INSTALL libsubcmd_headers
CALL scripts/checksyscalls.sh
CHK kernel/kheaders_data.tar.xz
GEN kernel/kheaders_data.tar.xz
CC kernel/kheaders.o

[2] Modify a header without changing the file size

$ sed -i s/0xdeadbeef/0xfeedbeef/ include/linux/elfnote.h

[3] Rebuild kernel/kheaders_data.tar.xz

$ make -j$(nproc) kernel/kheaders.o
DESCEND objtool
INSTALL libsubcmd_headers
CALL scripts/checksyscalls.sh
CHK kernel/kheaders_data.tar.xz

kernel/kheaders_data.tar.xz is not updated if steps [1] - [3] are run
within the same minute.

The headers_md5 variable stores the MD5 hash of the 'ls -l' output
for all header files. This hash value is used to determine whether
kheaders_data.tar.xz needs to be rebuilt. However, 'ls -l' prints the
modification times with minute-level granularity. If a file is modified
within the same minute and its size remains the same, the MD5 hash does
not change.

To reliably detect file modifications, this commit rewrites
kernel/gen_kheaders.sh to output header dependencies to
kernel/.kheaders_data.tar.xz.cmd. Then, Make compares the timestamps
and reruns kernel/gen_kheaders.sh when necessary. This is the standard
mechanism used by Make and Kbuild.

Signed-off-by: Masahiro Yamada <masahiroy@kernel.org>

+63 -74
+2
kernel/.gitignore
··· 1 1 # SPDX-License-Identifier: GPL-2.0-only 2 2 /config_data 3 3 /kheaders.md5 4 + /kheaders-objlist 5 + /kheaders-srclist
+42 -5
kernel/Makefile
··· 158 158 $(obj)/config_data: $(KCONFIG_CONFIG) FORCE 159 159 $(call filechk,cat) 160 160 161 + # kheaders_data.tar.xz 161 162 $(obj)/kheaders.o: $(obj)/kheaders_data.tar.xz 162 163 163 - quiet_cmd_genikh = CHK $(obj)/kheaders_data.tar.xz 164 - cmd_genikh = $(CONFIG_SHELL) $(srctree)/kernel/gen_kheaders.sh $@ 165 - $(obj)/kheaders_data.tar.xz: FORCE 166 - $(call cmd,genikh) 164 + quiet_cmd_kheaders_data = GEN $@ 165 + cmd_kheaders_data = "$<" "$@" "$(obj)/kheaders-srclist" "$(obj)/kheaders-objlist" 166 + cmd_kheaders_data_dep = cat $(depfile) >> $(dot-target).cmd; rm -f $(depfile) 167 167 168 - clean-files := kheaders_data.tar.xz kheaders.md5 168 + define rule_kheaders_data 169 + $(call cmd_and_savecmd,kheaders_data) 170 + $(call cmd,kheaders_data_dep) 171 + endef 172 + 173 + targets += kheaders_data.tar.xz 174 + $(obj)/kheaders_data.tar.xz: $(src)/gen_kheaders.sh $(obj)/kheaders-srclist $(obj)/kheaders-objlist $(obj)/kheaders.md5 FORCE 175 + $(call if_changed_rule,kheaders_data) 176 + 177 + # generated headers in objtree 178 + # 179 + # include/generated/utsversion.h is ignored because it is generated 180 + # after gen_kheaders.sh is executed. (utsversion.h is unneeded for kheaders) 181 + filechk_kheaders_objlist = \ 182 + for d in include "arch/$(SRCARCH)/include"; do \ 183 + find "$${d}/generated" ! -path "include/generated/utsversion.h" -a -name "*.h" -print; \ 184 + done 185 + 186 + $(obj)/kheaders-objlist: FORCE 187 + $(call filechk,kheaders_objlist) 188 + 189 + # non-generated headers in srctree 190 + filechk_kheaders_srclist = \ 191 + for d in include "arch/$(SRCARCH)/include"; do \ 192 + find "$(srctree)/$${d}" -path "$(srctree)/$${d}/generated" -prune -o -name "*.h" -print; \ 193 + done 194 + 195 + $(obj)/kheaders-srclist: FORCE 196 + $(call filechk,kheaders_srclist) 197 + 198 + # Some files are symlinks. If symlinks are changed, kheaders_data.tar.xz should 199 + # be rebuilt. 200 + filechk_kheaders_md5sum = xargs -r -a $< stat -c %N | md5sum 201 + 202 + $(obj)/kheaders.md5: $(obj)/kheaders-srclist FORCE 203 + $(call filechk,kheaders_md5sum) 204 + 205 + clean-files := kheaders.md5 kheaders-srclist kheaders-objlist
+19 -69
kernel/gen_kheaders.sh
··· 4 4 # This script generates an archive consisting of kernel headers 5 5 # for CONFIG_IKHEADERS. 6 6 set -e 7 - sfile="$(readlink -f "$0")" 8 - outdir="$(pwd)" 9 7 tarfile=$1 10 - tmpdir=$outdir/${tarfile%/*}/.tmp_dir 8 + srclist=$2 9 + objlist=$3 11 10 12 - dir_list=" 13 - include/ 14 - arch/$SRCARCH/include/ 15 - " 11 + dir=$(dirname "${tarfile}") 12 + tmpdir=${dir}/.tmp_dir 13 + depfile=${dir}/.$(basename "${tarfile}").d 16 14 17 - # Support incremental builds by skipping archive generation 18 - # if timestamps of files being archived are not changed. 15 + # generate dependency list. 16 + { 17 + echo 18 + echo "deps_${tarfile} := \\" 19 + sed 's:\(.*\): \1 \\:' "${srclist}" 20 + sed -n '/^include\/generated\/autoconf\.h$/!s:\(.*\): \1 \\:p' "${objlist}" 21 + echo 22 + echo "${tarfile}: \$(deps_${tarfile})" 23 + echo 24 + echo "\$(deps_${tarfile}):" 19 25 20 - # This block is useful for debugging the incremental builds. 21 - # Uncomment it for debugging. 22 - # if [ ! -f /tmp/iter ]; then iter=1; echo 1 > /tmp/iter; 23 - # else iter=$(($(cat /tmp/iter) + 1)); echo $iter > /tmp/iter; fi 24 - # find $all_dirs -name "*.h" | xargs ls -l > /tmp/ls-$iter 25 - 26 - all_dirs= 27 - if [ "$building_out_of_srctree" ]; then 28 - for d in $dir_list; do 29 - all_dirs="$all_dirs $srctree/$d" 30 - done 31 - fi 32 - all_dirs="$all_dirs $dir_list" 33 - 34 - # include/generated/utsversion.h is ignored because it is generated after this 35 - # script is executed. (utsversion.h is unneeded for kheaders) 36 - # 37 - # When Kconfig regenerates include/generated/autoconf.h, its timestamp is 38 - # updated, but the contents might be still the same. When any CONFIG option is 39 - # changed, Kconfig touches the corresponding timestamp file include/config/*. 40 - # Hence, the md5sum detects the configuration change anyway. We do not need to 41 - # check include/generated/autoconf.h explicitly. 42 - # 43 - # Ignore them for md5 calculation to avoid pointless regeneration. 44 - headers_md5="$(find $all_dirs -name "*.h" -a \ 45 - ! -path include/generated/utsversion.h -a \ 46 - ! -path include/generated/autoconf.h | 47 - xargs ls -l | md5sum | cut -d ' ' -f1)" 48 - 49 - # Any changes to this script will also cause a rebuild of the archive. 50 - this_file_md5="$(ls -l $sfile | md5sum | cut -d ' ' -f1)" 51 - if [ -f $tarfile ]; then tarfile_md5="$(md5sum $tarfile | cut -d ' ' -f1)"; fi 52 - if [ -f kernel/kheaders.md5 ] && 53 - [ "$(head -n 1 kernel/kheaders.md5)" = "$headers_md5" ] && 54 - [ "$(head -n 2 kernel/kheaders.md5 | tail -n 1)" = "$this_file_md5" ] && 55 - [ "$(tail -n 1 kernel/kheaders.md5)" = "$tarfile_md5" ]; then 56 - exit 57 - fi 58 - 59 - echo " GEN $tarfile" 26 + } > "${depfile}" 60 27 61 28 rm -rf "${tmpdir}" 62 29 mkdir "${tmpdir}" 63 30 64 - if [ "$building_out_of_srctree" ]; then 65 - ( 66 - cd $srctree 67 - for f in $dir_list 68 - do find "$f" -name "*.h"; 69 - done | tar -c -f - -T - | tar -xf - -C "${tmpdir}" 70 - ) 71 - fi 72 - 73 - for f in $dir_list; 74 - do find "$f" -name "*.h"; 75 - done | tar -c -f - -T - | tar -xf - -C "${tmpdir}" 76 - 77 - # Always exclude include/generated/utsversion.h 78 - # Otherwise, the contents of the tarball may vary depending on the build steps. 79 - rm -f "${tmpdir}/include/generated/utsversion.h" 31 + # shellcheck disable=SC2154 # srctree is passed as an env variable 32 + sed "s:^${srctree}/::" "${srclist}" | tar -c -f - -C "${srctree}" -T - | tar -xf - -C "${tmpdir}" 33 + tar -c -f - -T "${objlist}" | tar -xf - -C "${tmpdir}" 80 34 81 35 # Remove comments except SDPX lines 82 36 # Use a temporary file to store directory contents to prevent find/xargs from ··· 45 91 tar "${KBUILD_BUILD_TIMESTAMP:+--mtime=$KBUILD_BUILD_TIMESTAMP}" \ 46 92 --owner=0 --group=0 --sort=name --numeric-owner --mode=u=rw,go=r,a+X \ 47 93 -I $XZ -cf $tarfile -C "${tmpdir}/" . > /dev/null 48 - 49 - echo $headers_md5 > kernel/kheaders.md5 50 - echo "$this_file_md5" >> kernel/kheaders.md5 51 - echo "$(md5sum $tarfile | cut -d ' ' -f1)" >> kernel/kheaders.md5 52 94 53 95 rm -rf "${tmpdir}"