blob: 26d121816dd81420f9d1829d03a281dcfc8bfd2d [file] [log] [blame]
Wolfgang Denk4d1d9a32006-02-21 17:33:04 +01001#!/bin/sh
Rasmus Villemoesf62d0462020-08-25 17:28:39 +02002# SPDX-License-Identifier: GPL-2.0
Kyle Moffett787fb372011-12-16 11:14:40 +00003#
4# This scripts adds local version information from the version
Rasmus Villemoes86354212024-05-15 09:36:12 +02005# control system git.
Kyle Moffett787fb372011-12-16 11:14:40 +00006#
Masahiro Yamada422633e2014-08-22 19:42:28 +09007# If something goes wrong, send a mail the kernel build mailinglist
8# (see MAINTAINERS) and CC Nico Schottelius
9# <nico-linuxsetlocalversion -at- schottelius.org>.
10#
Kyle Moffett787fb372011-12-16 11:14:40 +000011#
Wolfgang Denk4d1d9a32006-02-21 17:33:04 +010012
13usage() {
Rasmus Villemoes86354212024-05-15 09:36:12 +020014 echo "Usage: $0 [--no-local] [srctree]" >&2
Wolfgang Denk4d1d9a32006-02-21 17:33:04 +010015 exit 1
16}
17
Rasmus Villemoes86354212024-05-15 09:36:12 +020018no_local=false
19if test "$1" = "--no-local"; then
20 no_local=true
Kyle Moffett787fb372011-12-16 11:14:40 +000021 shift
22fi
Rasmus Villemoes86354212024-05-15 09:36:12 +020023
24srctree=.
Kyle Moffett787fb372011-12-16 11:14:40 +000025if test $# -gt 0; then
26 srctree=$1
27 shift
28fi
29if test $# -gt 0 -o ! -d "$srctree"; then
30 usage
31fi
32
33scm_version()
34{
Rasmus Villemoes86354212024-05-15 09:36:12 +020035 local short=false
36 local no_dirty=false
37 local tag
38
39 while [ $# -gt 0 ];
40 do
41 case "$1" in
42 --short)
43 short=true;;
44 --no-dirty)
45 no_dirty=true;;
46 esac
47 shift
48 done
Kyle Moffett787fb372011-12-16 11:14:40 +000049
50 cd "$srctree"
Marek Vasutb2970c02025-03-02 19:30:28 +010051 if test -e .scmversion; then
52 cat .scmversion
53 return
54 fi
Rasmus Villemoes86354212024-05-15 09:36:12 +020055
56 if test -n "$(git rev-parse --show-cdup 2>/dev/null)"; then
Kyle Moffett787fb372011-12-16 11:14:40 +000057 return
58 fi
Rasmus Villemoes86354212024-05-15 09:36:12 +020059
60 if ! head=$(git rev-parse --verify HEAD 2>/dev/null); then
61 return
Kyle Moffett787fb372011-12-16 11:14:40 +000062 fi
63
Rasmus Villemoes86354212024-05-15 09:36:12 +020064 # mainline kernel: 6.2.0-rc5 -> v6.2-rc5
65 # stable kernel: 6.1.7 -> v6.1.7
66 version_tag=v$(echo "${KERNELVERSION}" | sed -E 's/^([0-9]+\.[0-9]+)\.0(.*)$/\1\2/')
Kyle Moffett787fb372011-12-16 11:14:40 +000067
Rasmus Villemoes86354212024-05-15 09:36:12 +020068 # If a localversion* file exists, and the corresponding
69 # annotated tag exists and is an ancestor of HEAD, use
70 # it. This is the case in linux-next.
71 tag=${file_localversion#-}
72 desc=
73 if [ -n "${tag}" ]; then
74 desc=$(git describe --match=$tag 2>/dev/null)
75 fi
Kyle Moffett787fb372011-12-16 11:14:40 +000076
Rasmus Villemoes86354212024-05-15 09:36:12 +020077 # Otherwise, if a localversion* file exists, and the tag
78 # obtained by appending it to the tag derived from
79 # KERNELVERSION exists and is an ancestor of HEAD, use
80 # it. This is e.g. the case in linux-rt.
81 if [ -z "${desc}" ] && [ -n "${file_localversion}" ]; then
82 tag="${version_tag}${file_localversion}"
83 desc=$(git describe --match=$tag 2>/dev/null)
84 fi
Kyle Moffett787fb372011-12-16 11:14:40 +000085
Rasmus Villemoes86354212024-05-15 09:36:12 +020086 # Otherwise, default to the annotated tag derived from KERNELVERSION.
87 if [ -z "${desc}" ]; then
88 tag="${version_tag}"
89 desc=$(git describe --match=$tag 2>/dev/null)
90 fi
Kyle Moffett787fb372011-12-16 11:14:40 +000091
Rasmus Villemoes86354212024-05-15 09:36:12 +020092 # If we are at the tagged commit, we ignore it because the version is
93 # well-defined.
94 if [ "${tag}" != "${desc}" ]; then
Kyle Moffett787fb372011-12-16 11:14:40 +000095
Rasmus Villemoes86354212024-05-15 09:36:12 +020096 # If only the short version is requested, don't bother
97 # running further git commands
98 if $short; then
99 echo "+"
100 return
Kyle Moffett787fb372011-12-16 11:14:40 +0000101 fi
Rasmus Villemoes86354212024-05-15 09:36:12 +0200102 # If we are past the tagged commit, we pretty print it.
103 # (like 6.1.0-14595-g292a089d78d3)
104 if [ -n "${desc}" ]; then
105 echo "${desc}" | awk -F- '{printf("-%05d", $(NF-1))}'
Kyle Moffett787fb372011-12-16 11:14:40 +0000106 fi
107
Rasmus Villemoes86354212024-05-15 09:36:12 +0200108 # Add -g and exactly 12 hex chars.
109 printf '%s%s' -g "$(echo $head | cut -c1-12)"
110 fi
Kyle Moffett787fb372011-12-16 11:14:40 +0000111
Rasmus Villemoes86354212024-05-15 09:36:12 +0200112 if ${no_dirty}; then
Kyle Moffett787fb372011-12-16 11:14:40 +0000113 return
Wolfgang Denk4d1d9a32006-02-21 17:33:04 +0100114 fi
Mike Frysingerbfac87a2008-04-08 14:00:57 -0400115
Rasmus Villemoes86354212024-05-15 09:36:12 +0200116 # Check for uncommitted changes.
117 # This script must avoid any write attempt to the source tree, which
118 # might be read-only.
119 # You cannot use 'git describe --dirty' because it tries to create
120 # .git/index.lock .
121 # First, with git-status, but --no-optional-locks is only supported in
122 # git >= 2.14, so fall back to git-diff-index if it fails. Note that
123 # git-diff-index does not refresh the index, so it may give misleading
124 # results.
125 # See git-update-index(1), git-diff-index(1), and git-status(1).
126 if {
127 git --no-optional-locks status -uno --porcelain 2>/dev/null ||
128 git diff-index --name-only HEAD
129 } | read dummy; then
130 printf '%s' -dirty
Kyle Moffett787fb372011-12-16 11:14:40 +0000131 fi
132}
133
134collect_files()
135{
Rasmus Villemoesf62d0462020-08-25 17:28:39 +0200136 local file res=
Kyle Moffett787fb372011-12-16 11:14:40 +0000137
138 for file; do
139 case "$file" in
140 *\~*)
141 continue
142 ;;
143 esac
144 if test -e "$file"; then
145 res="$res$(cat "$file")"
146 fi
147 done
148 echo "$res"
149}
150
Rasmus Villemoes86354212024-05-15 09:36:12 +0200151if [ -z "${KERNELVERSION}" ]; then
152 echo "KERNELVERSION is not set" >&2
Masahiro Yamada422633e2014-08-22 19:42:28 +0900153 exit 1
154fi
Kyle Moffett787fb372011-12-16 11:14:40 +0000155
156# localversion* files in the build and source directory
Rasmus Villemoes86354212024-05-15 09:36:12 +0200157file_localversion="$(collect_files localversion*)"
Kyle Moffett787fb372011-12-16 11:14:40 +0000158if test ! "$srctree" -ef .; then
Rasmus Villemoes86354212024-05-15 09:36:12 +0200159 file_localversion="${file_localversion}$(collect_files "$srctree"/localversion*)"
160fi
161
162if ${no_local}; then
163 echo "${KERNELVERSION}$(scm_version --no-dirty)"
164 exit 0
165fi
166
167if ! test -e include/config/auto.conf; then
168 echo "Error: kernelrelease not valid - run 'make prepare' to update it" >&2
169 exit 1
Kyle Moffett787fb372011-12-16 11:14:40 +0000170fi
171
Rasmus Villemoes86354212024-05-15 09:36:12 +0200172# version string from CONFIG_LOCALVERSION
173config_localversion=$(sed -n 's/^CONFIG_LOCALVERSION=\(.*\)$/\1/p' include/config/auto.conf | tr -d '"')
Kyle Moffett787fb372011-12-16 11:14:40 +0000174
Rasmus Villemoes86354212024-05-15 09:36:12 +0200175# scm version string if not at the kernel version tag or at the file_localversion
176if grep -q "^CONFIG_LOCALVERSION_AUTO=y$" include/config/auto.conf; then
Kyle Moffett787fb372011-12-16 11:14:40 +0000177 # full scm version string
Rasmus Villemoes86354212024-05-15 09:36:12 +0200178 scm_version="$(scm_version)"
179elif [ "${LOCALVERSION+set}" != "set" ]; then
180 # If the variable LOCALVERSION is not set, append a plus
181 # sign if the repository is not in a clean annotated or
182 # signed tagged state (as git describe only looks at signed
183 # or annotated tags - git tag -a/-s).
184 #
185 # If the variable LOCALVERSION is set (including being set
186 # to an empty string), we don't want to append a plus sign.
187 scm_version="$(scm_version --short)"
Wolfgang Denk4d1d9a32006-02-21 17:33:04 +0100188fi
Mike Frysingerbfac87a2008-04-08 14:00:57 -0400189
Rasmus Villemoes86354212024-05-15 09:36:12 +0200190echo "${KERNELVERSION}${file_localversion}${config_localversion}${LOCALVERSION}${scm_version}"