#!/bin/bash # kmodtool - Helper script for building kernel module RPMs # Copyright (c) 2003-2012 Ville Skyttä , # Thorsten Leemhuis # Copyright (c) 2012-2019 Nicolas Chauvet # Copyright (c) 2017-2018 Nicolas Viéville # # Permission is hereby granted, free of charge, to any person obtaining # a copy of this software and associated documentation files (the # "Software"), to deal in the Software without restriction, including # without limitation the rights to use, copy, modify, merge, publish, # distribute, sublicense, and/or sell copies of the Software, and to # permit persons to whom the Software is furnished to do so, subject to # the following conditions: # # The above copyright notice and this permission notice shall be # included in all copies or substantial portions of the Software. # # THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, # EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF # MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND # NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE # LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION # OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION # WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. shopt -s extglob myprog="kmodtool-${repo}" myver="0.12.1" kmodname= build_kernels="current" kernels_known_variants= kernel_versions= kernel_versions_to_build_for= filterfile= target= kernel_uname_r="" kernel_uname_r_vanilla="" kernel_uname_r_short="" kernel_uname_r_wo_arch="" kernel_uname_r_dist="" kernel_uname_r_rel="" kernel_uname_r_rel_plus_one="" kernel_uname_r_short_plus_one="" kernel_uname_r_rel_minor="" error_out() { local errorlevel=${1} shift echo "Error: $@" >&2 # the next line is not multi-line safe -- not needed *yet* echo "%global kmodtool_check echo \"kmodtool error: $@\"; exit ${errorlevel};" exit ${errorlevel} } print_rpmtemplate_header() { echo echo '%global kmodinstdir_prefix '${usr_lib_modules}/ echo '%global kmodinstdir_postfix '/extra/${kmodname}/ echo '%global kernel_versions '${kernel_versions} echo } init_kernel_uname_r_vars() { ## Build some interesting strings with kernel uname -r ## Examples: ## - Fedora and rhel: ## kernel_uname_r 3.10.0-693.2.2.el7.x86_64 ## kernel_uname_r_vanilla 3.10.0 ## kernel_uname_r_short 3.10.0-693.el7 ## kernel_uname_r_wo_arch 3.10.0-693.2.2.el7 ## kernel_uname_r_dist el7 ## kernel_uname_r_rel 693 ## kernel_uname_r_rel_plus_one 694 ## kernel_uname_r_short_plus_one 3.10.0-694.el7 ## kernel_uname_r_rel_minor 2.2 ## kernel_uname_r=${1} kernel_uname_r_vanilla=$(echo ${1} | sed -n -e "s/\.${target}//" -e "s/\([[:alnum:]\.]\+\|\)-\(.*\)/\1/p") kernel_uname_r_short=$(echo ${1} | sed -n -e "s/\.${target}//" -e "s/\(.*\)-\([[:alnum:]]\+\)\(\.[[:alnum:]\.]\+\|\)\(\.el.*\|\.fc.*\)/\1-\2\4/p") kernel_uname_r_wo_arch=$(echo ${1} | sed -e "s/\.${target}//") kernel_uname_r_dist=$(echo ${1} | sed -n -e "s/\.${target}//" -e "s/\(.*\)-\([[:alnum:]]\+\.[[:alnum:]]\+\-\|[[:alnum:]]\+\)\([[:alnum:]\.]\+\|\)\(\.el.*\|\.fc.*\)/\4/" -e "s/^\.//p") kernel_uname_r_rel=$(echo ${1} | sed -n -e "s/\.${target}//" -e "s/\(.*\)-\([[:alnum:]]\+\.[[:alnum:]]\+\-\|[[:alnum:]]\+\.\)\([[:alnum:]\.]\+\|\)\(.*\)/\2/" -e "s/^\.//" -e "s/[\.\-]$//p") if $(echo "${kernel_uname_r_rel}" | grep -qE '^[0-9]+$') ; then kernel_uname_r_rel_plus_one=$(( kernel_uname_r_rel+1 )) else kernel_uname_r_rel_plus_one="" fi kernel_uname_r_short_plus_one=$(echo ${kernel_uname_r_short} | sed -e "s/(//" -e "s/-${kernel_uname_r_rel}\./-${kernel_uname_r_rel_plus_one}\./") kernel_uname_r_rel_minor=$(echo ${1} | sed -n -e "s/\.${target}//" -e "s/\(.*\)-\([[:alnum:]]\+\.[[:alnum:]]\+\-\|[[:alnum:]]\+\.\)\([[:alnum:]\.]\+\|\)\(\.el.*\|\.fc.*\)/\3/" -e "s/^\.//p") } print_rpmtemplate_macros() { # Defines package names macros depending on distributions (default # is Fedora): # - rhel_kabi is set if building for RHEL # - kmod_pkg_name and pkg_kmod_name are the two possible forms of # the package name used when building packages modules. First one # uses kmod as prefix, the second one uses kmod as suffix. These # macros allow to modify the naming schema regarding the target # distribution (for example with suffix and prefix different from # kmod). # - kmod_obsolete_name same as kmod_pkg_name and pkg_kmod_name but # for obsolete package name # if [[ -n ${obsolete_name} ]] ; then echo '%global obsolete_name '${obsolete_name} fi cat <= %{?epoch:%{epoch}:}%{version} Provides: %{pkg_kmod_name} = %{?epoch:%{epoch}:}%{version}-%{release} EOF if [[ -n "${obsolete_name}" ]] ; then echo "Provides: akmod-${obsolete_name} = ${obsolete_version}" echo "Obsoletes: akmod-${obsolete_name} < ${obsolete_version}" fi cat < /dev/null & %post -n akmod-${kmodname} [ -x /usr/sbin/akmods-ostree-post ] && /usr/sbin/akmods-ostree-post ${kmodname} %{_usrsrc}/akmods/%{pkg_kmod_name}-%{version}-%{release}.src.rpm %files -n akmod-${kmodname} %defattr(-,root,root,-) %{_usrsrc}/akmods/* EOF } print_akmodmeta () { cat <= %{?epoch:%{epoch}:}%{version} Requires(post): %{?_prefix_usr}/sbin/depmod Requires(postun): %{?_prefix_usr}/sbin/depmod %if %{with rhel_kabi} Requires(post): %{?_prefix_usr}/sbin/weak-modules Requires(postun): %{?_prefix_usr}/sbin/weak-modules %endif EOF if [[ -n "${obsolete_name}" ]] ; then echo echo "Provides: %{kmod_obsolete_name}-${kernel_uname_r} = ${obsolete_version}" echo "Obsoletes: %{kmod_obsolete_name}-${kernel_uname_r} < ${obsolete_version}" fi # second part if [[ ! -n "${customkernel}" ]] ; then cat </dev/null ; \\\ elif [ -f /lib/modules/${kernel_uname_r}/System.map ] ; then \\\ %{?_prefix_usr}/sbin/depmod -aeF /lib/modules/${kernel_uname_r}/System.map ${kernel_uname_r} >/dev/null ; \\\ else \\\ %{?_prefix_usr}/sbin/depmod -a >/dev/null ; \\\ fi || : %if %{with rhel_kabi} Requires: kernel >= ${kernel_uname_r_short} Requires: kernel < ${kernel_uname_r_short_plus_one} BuildRequires: kernel-devel = ${kernel_uname_r_wo_arch} redhat-rpm-config %else Requires: kernel-uname-r = ${kernel_uname_r} BuildRequires: kernel-devel-uname-r = ${kernel_uname_r} BuildRequires: gcc %endif %post -n %{package_kmod_name} %{_kmodtool_depmod_post} %if %{with rhel_kabi} modules=( \$(find ${usr_lib_modules}/${kernel_uname_r}/extra/${kmodname}/ -name "*.ko" -o -name "*.ko.xz") ) printf '%s\n' "\${modules[@]}" | %{?_prefix_usr}/sbin/weak-modules --add-modules %endif %if %{with rhel_kabi} %preun -n %{package_kmod_name} rpm -ql %{package_kmod_name}-%{version}-%{release}.$(arch) | grep -e '\.ko$' -e '\.ko.xz$' > /var/run/rpm-%{package_kmod_name}-modules %endif %postun -n %{package_kmod_name} %{_kmodtool_depmod_post} %if %{with rhel_kabi} modules=( \$(cat /var/run/rpm-%{package_kmod_name}-modules) ) rm -f /var/run/rpm-%{package_kmod_name}-modules printf '%s\n' "\${modules[@]}" | %{?_prefix_usr}/sbin/weak-modules --remove-modules %endif EOF else cat < /dev/null || : %if %{with rhel_kabi} if [[ "\$(uname -r)" == "${kernel_uname_r}" ]] ; then modules=( \$(find ${usr_lib_modules}/${kernel_uname_r}/extra/${kmodname}/ -name "*.ko" -o -name "*.ko.xz") ) printf '%s\n' "\${modules[@]}" | %{?_prefix_usr}/sbin/weak-modules --add-modules fi %endif %if %{with rhel_kabi} %preun -n %{package_kmod_name} if [[ "\$(uname -r)" == "${kernel_uname_r}" ]] ; then rpm -ql %{package_kmod_name}-%{version}-%{release}.$(arch) | grep -e '\.ko$' -e '\.ko.xz$' > /var/run/rpm-%{package_kmod_name}-modules fi %endif %postun -n %{package_kmod_name} [[ "\$(uname -r)" == "${kernel_uname_r}" ]] && %{?_prefix_usr}/sbin/depmod -a > /dev/null || : %if %{with rhel_kabi} if [[ "\$(uname -r)" == "${kernel_uname_r}" ]] ; then modules=( \$(cat /var/run/rpm-%{package_kmod_name}-modules) ) rm -f /var/run/rpm-%{package_kmod_name}-modules printf '%s\n' "\${modules[@]}" | %{?_prefix_usr}/sbin/weak-modules --remove-modules fi %endif EOF fi # third part cat <= %{?epoch:%{epoch}:}%{version}-%{release} EOF if [[ -n "${obsolete_name}" ]] ; then echo "Provides: %{kmod_obsolete_name}${kernel_variant} = ${obsolete_version}" echo "Obsoletes: %{kmod_obsolete_name}${kernel_variant} < ${obsolete_version}" fi cat < -- look for our shared files in " echo " --filterfile -- filter the results with grep --file " echo " --for-kernels -- created templates only for these kernels" echo " --kmodname -- name of the kmod (required)" echo " --noakmod -- no akmod package" echo " --repo -- use buildsys-build--kerneldevpkgs" echo " --target -- target-arch (required)" } while [ "${1}" ] ; do case "${1}" in --filterfile) shift if [[ ! -n "${1}" ]] ; then error_out 2 "Please provide path to a filter-file together with --filterfile" >&2 elif [[ ! -e "${1}" ]] ; then error_out 2 "Filterfile ${1} not found" >&2 fi filterfile="${1}" shift ;; --kmodname) shift if [[ ! -n "${1}" ]] ; then error_out 2 "Please provide the name of the kmod together with --kmodname" >&2 fi # strip pending -kmod kmodname="${1%%-kmod}" shift ;; --repo) shift if [[ ! -n "${1}" ]] ; then error_out 2 "Please provide the name of the repo together with --repo" >&2 fi repo=${1} shift ;; --for-kernels) shift if [[ ! -n "${1}" ]] ; then error_out 2 "Please provide the name of the kmod together with --kmodname" >&2 fi for_kernels="${1}" shift ;; --noakmod) shift noakmod="true" ;; --obsolete-name) shift if [[ ! -n "${1}" ]] ; then error_out 2 "Please provide the name of the kmod to obsolte together with --obsolete-name" >&2 fi obsolete_name="${1}" shift ;; --obsolete-version) shift if [[ ! -n "${1}" ]] ; then error_out 2 "Please provide the version of the kmod to obsolte together with --obsolete-version" >&2 fi obsolete_version="${1}" shift ;; --target) shift target="${1}" shift ;; --akmod) shift build_kernels="akmod" ;; --newest) shift build_kernels="newest" ;; --current) shift build_kernels="current" ;; --help) myprog_help exit 0 ;; --version) echo "${myprog} ${myver}" exit 0 ;; *) echo "Error: Unknown option '${1}'." >&2 usage >&2 exit 2 ;; esac done # These should always be in /lib/modules # as that's how the kernel package lays them out usr_lib_modules=/lib/modules if [[ -e ./kmodtool-kernel-variants ]] ; then kernels_known_variants="$(cat ./kmodtool-kernel-variants)" elif [[ -e /usr/share/kmodtool/kernel-variants ]] ; then kernels_known_variants="$(cat /usr/share/kmodtool/kernel-variants)" else error_out 2 "Could not find /usr/share/kmodtool/kernel-variants" fi # general sanity checks if [[ ! -n "${target}" ]] ; then error_out 2 "please pass target arch with --target" elif [[ ! -n "${kmodname}" ]] ; then error_out 2 "please pass kmodname with --kmodname" elif [[ ! -n "${kernels_known_variants}" ]] ; then error_out 2 "could not determine known variants" elif ( [[ -n "${obsolete_name}" ]] && [[ ! -n "${obsolete_version}" ]] ) || ( [[ ! -n "${obsolete_name}" ]] && [[ -n "${obsolete_version}" ]] ) ; then error_out 2 "you need to provide both --obsolete-name and --obsolete-version" fi # go print_rpmtemplate_macros if [[ -n "${for_kernels}" ]] ; then # this is easy: print_customrpmtemplate "${for_kernels}" elif [[ "${build_kernels}" == "akmod" ]] ; then # do only a akmod package print_akmodtemplate print_akmodmeta else # seems we are on out own to decide for which kernels to build # we need more sanity checks in this case if [[ ! -n "${repo}" ]] ; then error_out 2 "please provide repo name with --repo" elif ! $(which buildsys-build-${repo}-kerneldevpkgs &> /dev/null) ; then error_out 2 "buildsys-build-${repo}-kerneldevpkgs not found" fi # call buildsys-build-${repo}-kerneldevpkgs to get the list of kernels cmdoptions="--target ${target}" # filterfile to filter list of kernels? if [[ -n "${filterfile}" ]] ; then cmdoptions="${cmdoptions} --filterfile ${filterfile}" fi kernel_versions_to_build_for="$(buildsys-build-${repo}-kerneldevpkgs --${build_kernels} ${cmdoptions})" returncode=$? if (( ${returncode} != 0 )) ; then error_out 2 "buildsys-build-${repo}-kerneldevpkgs failed: $(buildsys-build-${repo}-kerneldevpkgs --${build_kernels} ${cmdoptions})" fi if [[ "${build_kernels}" == "current" ]] && [[ ! -n "${noakmod}" ]] ; then print_akmodtemplate fi print_rpmtemplate fi