xref: /linux/scripts/setlocalversion (revision a2be76a352f1035a2e5f914a409743d65dc514c5)
1117a93dbSRene Scharfe#!/bin/sh
2b2441318SGreg Kroah-Hartman# SPDX-License-Identifier: GPL-2.0
333252572SNico Schottelius#
433252572SNico Schottelius# This scripts adds local version information from the version
533252572SNico Schottelius# control systems git, mercurial (hg) and subversion (svn).
633252572SNico Schottelius#
733252572SNico Schottelius# If something goes wrong, send a mail the kernel build mailinglist
833252572SNico Schottelius# (see MAINTAINERS) and CC Nico Schottelius
933252572SNico Schottelius# <nico-linuxsetlocalversion -at- schottelius.org>.
1033252572SNico Schottelius#
1133252572SNico Schottelius#
12aaebf433SRyan Anderson
13117a93dbSRene Scharfeusage() {
14b003afe3SMichal Marek	echo "Usage: $0 [--save-scmversion] [srctree]" >&2
15117a93dbSRene Scharfe	exit 1
16aaebf433SRyan Anderson}
17aaebf433SRyan Anderson
1809155120SMichal Marekscm_only=false
1909155120SMichal Mareksrctree=.
20b003afe3SMichal Marekif test "$1" = "--save-scmversion"; then
2109155120SMichal Marek	scm_only=true
2209155120SMichal Marek	shift
2309155120SMichal Marekfi
2409155120SMichal Marekif test $# -gt 0; then
2509155120SMichal Marek	srctree=$1
2609155120SMichal Marek	shift
2709155120SMichal Marekfi
2809155120SMichal Marekif test $# -gt 0 -o ! -d "$srctree"; then
2909155120SMichal Marek	usage
3009155120SMichal Marekfi
3109155120SMichal Marek
3209155120SMichal Marekscm_version()
3309155120SMichal Marek{
346dc0c2f3SMichał Górny	local short
356dc0c2f3SMichał Górny	short=false
3609155120SMichal Marek
3709155120SMichal Marek	cd "$srctree"
3809155120SMichal Marek	if test -e .scmversion; then
396dc0c2f3SMichał Górny		cat .scmversion
4009155120SMichal Marek		return
4109155120SMichal Marek	fi
4209155120SMichal Marek	if test "$1" = "--short"; then
4309155120SMichal Marek		short=true
4409155120SMichal Marek	fi
45aaebf433SRyan Anderson
46117a93dbSRene Scharfe	# Check for git and a git repo.
477593e090SFranck Bui-Huu	if test -z "$(git rev-parse --show-cdup 2>/dev/null)" &&
48548b8b51SRasmus Villemoes	   head=$(git rev-parse --verify HEAD 2>/dev/null); then
4933252572SNico Schottelius
5009155120SMichal Marek		# If we are at a tagged commit (like "v2.6.30-rc6"), we ignore
5109155120SMichal Marek		# it, because this version is defined in the top level Makefile.
523c96bdd0SBhaskar Chowdhury		if [ -z "$(git describe --exact-match 2>/dev/null)" ]; then
5333252572SNico Schottelius
5409155120SMichal Marek			# If only the short version is requested, don't bother
5509155120SMichal Marek			# running further git commands
5609155120SMichal Marek			if $short; then
5709155120SMichal Marek				echo "+"
5809155120SMichal Marek				return
5909155120SMichal Marek			fi
6009155120SMichal Marek			# If we are past a tagged commit (like
6109155120SMichal Marek			# "v2.6.30-rc5-302-g72357d5"), we pretty print it.
62548b8b51SRasmus Villemoes			#
63548b8b51SRasmus Villemoes			# Ensure the abbreviated sha1 has exactly 12
64548b8b51SRasmus Villemoes			# hex characters, to make the output
65548b8b51SRasmus Villemoes			# independent of git version, local
66548b8b51SRasmus Villemoes			# core.abbrev settings and/or total number of
67548b8b51SRasmus Villemoes			# objects in the current repository - passing
68548b8b51SRasmus Villemoes			# --abbrev=12 ensures a minimum of 12, and the
69548b8b51SRasmus Villemoes			# awk substr() then picks the 'g' and first 12
70548b8b51SRasmus Villemoes			# hex chars.
71548b8b51SRasmus Villemoes			if atag="$(git describe --abbrev=12 2>/dev/null)"; then
72548b8b51SRasmus Villemoes				echo "$atag" | awk -F- '{printf("-%05d-%s", $(NF-1),substr($(NF),0,13))}'
7333252572SNico Schottelius
74548b8b51SRasmus Villemoes			# If we don't have a tag at all we print -g{commitish},
75548b8b51SRasmus Villemoes			# again using exactly 12 hex chars.
76f03b283fSTrent Piepho			else
77548b8b51SRasmus Villemoes				head="$(echo $head | cut -c1-12)"
78f03b283fSTrent Piepho				printf '%s%s' -g $head
7956b2f070SSebastian Siewior			fi
8033252572SNico Schottelius		fi
81aaebf433SRyan Anderson
82ff64dd48SBrian Norris		# Check for uncommitted changes.
83ff64dd48SBrian Norris		# First, with git-status, but --no-optional-locks is only
84ff64dd48SBrian Norris		# supported in git >= 2.14, so fall back to git-diff-index if
85ff64dd48SBrian Norris		# it fails. Note that git-diff-index does not refresh the
86ff64dd48SBrian Norris		# index, so it may give misleading results. See
87ff64dd48SBrian Norris		# git-update-index(1), git-diff-index(1), and git-status(1).
88ff64dd48SBrian Norris		if {
89ff64dd48SBrian Norris			git --no-optional-locks status -uno --porcelain 2>/dev/null ||
90ff64dd48SBrian Norris			git diff-index --name-only HEAD
91*a2be76a3SMasahiro Yamada		} | read dummy; then
9224d49756SRyan Anderson			printf '%s' -dirty
93117a93dbSRene Scharfe		fi
9409155120SMichal Marek	fi
9509155120SMichal Marek}
9609155120SMichal Marek
9709155120SMichal Marekcollect_files()
9809155120SMichal Marek{
997a82e3faSMasahiro Yamada	local file res=
10009155120SMichal Marek
10109155120SMichal Marek	for file; do
10209155120SMichal Marek		case "$file" in
10309155120SMichal Marek		*\~*)
10409155120SMichal Marek			continue
10509155120SMichal Marek			;;
10609155120SMichal Marek		esac
10709155120SMichal Marek		if test -e "$file"; then
10809155120SMichal Marek			res="$res$(cat "$file")"
10909155120SMichal Marek		fi
11009155120SMichal Marek	done
11109155120SMichal Marek	echo "$res"
11209155120SMichal Marek}
11309155120SMichal Marek
11409155120SMichal Marekif $scm_only; then
115b003afe3SMichal Marek	if test ! -e .scmversion; then
116b003afe3SMichal Marek		res=$(scm_version)
117b003afe3SMichal Marek		echo "$res" >.scmversion
118b003afe3SMichal Marek	fi
119ba3d05fbSBryan Wu	exit
120ba3d05fbSBryan Wufi
12109155120SMichal Marek
12209155120SMichal Marekif test -e include/config/auto.conf; then
1236dc0c2f3SMichał Górny	. include/config/auto.conf
12409155120SMichal Marekelse
12578283edfSWolfram Sang	echo "Error: kernelrelease not valid - run 'make prepare' to update it" >&2
12609155120SMichal Marek	exit 1
12709155120SMichal Marekfi
12809155120SMichal Marek
12909155120SMichal Marek# localversion* files in the build and source directory
13009155120SMichal Marekres="$(collect_files localversion*)"
13109155120SMichal Marekif test ! "$srctree" -ef .; then
13209155120SMichal Marek	res="$res$(collect_files "$srctree"/localversion*)"
13309155120SMichal Marekfi
13409155120SMichal Marek
13509155120SMichal Marek# CONFIG_LOCALVERSION and LOCALVERSION (if set)
13609155120SMichal Marekres="${res}${CONFIG_LOCALVERSION}${LOCALVERSION}"
13709155120SMichal Marek
13809155120SMichal Marek# scm version string if not at a tagged commit
13909155120SMichal Marekif test "$CONFIG_LOCALVERSION_AUTO" = "y"; then
14009155120SMichal Marek	# full scm version string
14109155120SMichal Marek	res="$res$(scm_version)"
14209155120SMichal Marekelse
143c3e2f196SMichael Prokop	# append a plus sign if the repository is not in a clean
144c3e2f196SMichael Prokop	# annotated or signed tagged state (as git describe only
145c3e2f196SMichael Prokop	# looks at signed or annotated tags - git tag -a/-s) and
146c3e2f196SMichael Prokop	# LOCALVERSION= is not specified
14709155120SMichal Marek	if test "${LOCALVERSION+set}" != "set"; then
14809155120SMichal Marek		scm=$(scm_version --short)
14909155120SMichal Marek		res="$res${scm:++}"
15009155120SMichal Marek	fi
15109155120SMichal Marekfi
15209155120SMichal Marek
15309155120SMichal Marekecho "$res"
154