|  | #!/bin/bash | 
|  | # SPDX-License-Identifier: GPL-2.0 | 
|  |  | 
|  | # 2 args: | 
|  | #	libxfs-apply <repo> <commit ID or patchfile> | 
|  |  | 
|  | usage() | 
|  | { | 
|  | echo $* | 
|  | echo | 
|  | echo "Usage:" | 
|  | echo "	libxfs-apply [--verbose] --sob <name/email> --source <repodir> --commit <commit_id>" | 
|  | echo "	libxfs-apply --patch <patchfile>" | 
|  | echo | 
|  | echo "libxfs-apply should be run in the destination git repository." | 
|  | exit | 
|  | } | 
|  |  | 
|  | cleanup() | 
|  | { | 
|  | rm -f $PATCH | 
|  | } | 
|  |  | 
|  | # output to stderr so it is not caught by file redirects | 
|  | fail() | 
|  | { | 
|  | >&2 echo "Fail:" | 
|  | >&2 echo $* | 
|  | cleanup | 
|  | exit | 
|  | } | 
|  |  | 
|  | # filterdiff didn't start handling git diff metadata correctly until some time | 
|  | # after 0.3.4. The handling in 0.3.4 was buggy and broken, requiring working | 
|  | # around that bugs to use it. Now that 0.4.2 has fixed all those bugs, the | 
|  | # work-arounds for 0.3.4 do not work. Hence set 0.4.2 as the minimum required | 
|  | # version and tell the user to upgrade if an old version is detected. We need to | 
|  | # check against x.y.z version numbers here. | 
|  | _version=`filterdiff --version | cut -d " " -f 5` | 
|  | _major=`echo $_version | cut -d "." -f 1` | 
|  | _minor=`echo $_version | cut -d "." -f 2` | 
|  | _patch=`echo $_version | cut -d "." -f 3` | 
|  | if [ $_major -eq 0 ]; then | 
|  | if [ $_minor -lt 4 ]; then | 
|  | fail "filterdiff $_version found. 0.4.2 or greater is required." | 
|  | fi | 
|  | if [ $_minor -eq 4 -a $_patch -lt 2 ]; then | 
|  | fail "filterdiff $_version found. 0.4.2 or greater is required." | 
|  | fi | 
|  | fi | 
|  |  | 
|  | # We should see repository contents we recognise, both at the source and | 
|  | # destination. Kernel repositorys will have fs/xfs/libxfs, and xfsprogs | 
|  | # repositories will have libxcmd. | 
|  | SOURCE="kernel" | 
|  | check_repo() | 
|  | { | 
|  | if [ ! -d "fs/xfs/libxfs" -a ! -d "libxcmd" ]; then | 
|  | usage "$1 repository contents not recognised!" | 
|  | fi | 
|  | if [ -d "$REPO/libxcmd" ]; then | 
|  | SOURCE="xfsprogs" | 
|  | fi | 
|  | } | 
|  |  | 
|  | REPO= | 
|  | PATCH= | 
|  | COMMIT_ID= | 
|  | VERBOSE= | 
|  | GUILT=0 | 
|  | STGIT=0 | 
|  |  | 
|  | while [ $# -gt 0 ]; do | 
|  | case "$1" in | 
|  | --source)	REPO=$2 ; shift ;; | 
|  | --patch)	PATCH=$2; shift ;; | 
|  | --commit)	COMMIT_ID=$2 ; shift ;; | 
|  | --sob)		SIGNED_OFF_BY=$2 ; shift ;; | 
|  | --verbose)	VERBOSE=true ;; | 
|  | *)		usage ;; | 
|  | esac | 
|  | shift | 
|  | done | 
|  |  | 
|  | if [ -n "$PATCH" ]; then | 
|  | if [ -n "$REPO" -o -n "$COMMIT_ID" ]; then | 
|  | usage "Need to specify either patch or source repo/commit" | 
|  | fi | 
|  | VERBOSE=true | 
|  | elif [ -z "$REPO" -o -z "$COMMIT_ID" ]; then | 
|  | usage "Need to specify both source repo and commit id" | 
|  | fi | 
|  |  | 
|  | check_repo Destination | 
|  |  | 
|  | # Are we using guilt? This works even if no patch is applied. | 
|  | guilt top &> /dev/null | 
|  | if [ $? -eq 0 ]; then | 
|  | GUILT=1 | 
|  | fi | 
|  |  | 
|  | # Are we using stgit? This works even if no patch is applied. | 
|  | stg top &> /dev/null | 
|  | if [ $? -eq 0 ]; then | 
|  | STGIT=1 | 
|  | fi | 
|  |  | 
|  | #this is pulled from the guilt code to handle commit ids sanely. | 
|  | # usage: munge_hash_range <hash range> | 
|  | # | 
|  | # this means: | 
|  | #	<hash>			- one commit | 
|  | #	<hash>..		- hash until head (excludes hash, includes head) | 
|  | #	..<hash>		- until hash (includes hash) | 
|  | #	<hash1>..<hash2>	- from hash to hash (inclusive) | 
|  | # | 
|  | # The output of this function is suitable to be passed to "git rev-list" | 
|  | munge_hash_range() | 
|  | { | 
|  | case "$1" in | 
|  | *..*..*|*\ *) | 
|  | # double .. or space is illegal | 
|  | return 1;; | 
|  | ..*) | 
|  | # e.g., "..v0.10" | 
|  | echo ${1#..};; | 
|  | *..) | 
|  | # e.g., "v0.19.." | 
|  | echo ${1%..}..HEAD;; | 
|  | *..*) | 
|  | # e.g., "v0.19-rc1..v0.19" | 
|  | echo ${1%%..*}..${1#*..};; | 
|  | ?*) | 
|  | # e.g., "v0.19" | 
|  | echo $1^..$1;; | 
|  | *)  # empty | 
|  | return 1;; | 
|  | esac | 
|  | return 0 | 
|  | } | 
|  |  | 
|  | # Filter the patch into the right format & files for the other tree | 
|  | filter_kernel_patch() | 
|  | { | 
|  | local _patch=$1 | 
|  | local _libxfs_files="" | 
|  |  | 
|  | # The files we will try to apply to | 
|  | _libxfs_files=`mktemp` | 
|  | ls -1 fs/xfs/libxfs/*.[ch] | sed -e "s%.*/\(.*\)%*\1%" > $_libxfs_files | 
|  |  | 
|  | # Create the new patch | 
|  | # filterdiff will have screwed up files that source/sink /dev/null. | 
|  | # fix that up with some sed magic. | 
|  | filterdiff \ | 
|  | --verbose \ | 
|  | -I $_libxfs_files \ | 
|  | --strip=1 \ | 
|  | --addoldprefix=a/fs/xfs/ \ | 
|  | --addnewprefix=b/fs/xfs/ \ | 
|  | $_patch | \ | 
|  | sed -e 's, [ab]\/fs\/xfs\/\(\/dev\/null\), \1,' | 
|  |  | 
|  |  | 
|  | rm -f $_libxfs_files | 
|  | } | 
|  |  | 
|  | filter_xfsprogs_patch() | 
|  | { | 
|  | local _patch=$1 | 
|  | local _libxfs_files="" | 
|  |  | 
|  | # The files we will try to apply to. We need to pull this from the | 
|  | # patch, as the may be libxfs files added in this patch and so we | 
|  | # need to capture them. | 
|  | _libxfs_files=`mktemp` | 
|  | #ls -1 libxfs/*.[ch] | sed -e "s%.*/\(.*\)%*libxfs/\1%" > $_libxfs_files | 
|  | lsdiff $_patch | sed -e "s%.*/\(.*\)%*libxfs/\1%" > $_libxfs_files | 
|  |  | 
|  | # Create the new patch | 
|  | # filterdiff will have screwed up files that source/sink /dev/null. | 
|  | # fix that up with some sed magic. | 
|  | filterdiff \ | 
|  | --verbose \ | 
|  | -I $_libxfs_files \ | 
|  | --strip=3 \ | 
|  | --addoldprefix=a/ \ | 
|  | --addnewprefix=b/ \ | 
|  | $_patch | \ | 
|  | sed -e 's, [ab]\/\(\/dev\/null\), \1,' | 
|  |  | 
|  | rm -f $_libxfs_files | 
|  | } | 
|  |  | 
|  | add_header() | 
|  | { | 
|  | local hdr="$1" | 
|  | local hdrfile="$2" | 
|  |  | 
|  | tail -n 1 "$hdrfile" | grep -q "^${hdr}$" || echo "$hdr" >> "$hdrfile" | 
|  | } | 
|  |  | 
|  | fixup_header_format() | 
|  | { | 
|  | local _source=$1 | 
|  | local _patch=$2 | 
|  | local _hdr=`mktemp` | 
|  | local _diff=`mktemp` | 
|  | local _new_hdr=$_hdr.new | 
|  |  | 
|  | # Split the header on the first ^diff --git line (convenient!) | 
|  | sed -e /^diff/q $_patch > $_hdr | 
|  | cat $_patch | awk ' | 
|  | BEGIN { diff_seen = 0; index_seen = 0 } | 
|  | /^diff/ { diff_seen++; next } | 
|  | /^index/ { if (++index_seen == 1) { next } } | 
|  | // { if (diff_seen) { print $0 } }' > $_diff | 
|  |  | 
|  | # the header now has the format: | 
|  | # commit 0d5a75e9e23ee39cd0d8a167393dcedb4f0f47b2 | 
|  | # Author: Eric Sandeen <sandeen@sandeen.net> | 
|  | # Date:   Wed Jun 1 17:38:15 2016 +1000 | 
|  | # | 
|  | #     xfs: make several functions static | 
|  | #.... | 
|  | #     Signed-off-by: Dave Chinner <david@fromorbit.com> | 
|  | # | 
|  | # We want to format it like a normal patch with a line to say what repo | 
|  | # and commit it was sourced from: | 
|  | # | 
|  | # xfs: make several functions static | 
|  | # | 
|  | # From: Eric Sandeen <sandeen@sandeen.net> | 
|  | # | 
|  | # Source kernel commit: 0d5a75e9e23ee39cd0d8a167393dcedb4f0f47b2 | 
|  | # | 
|  | # <body> | 
|  | # | 
|  | # To do this, use sed to first strip whitespace, then pass it into awk | 
|  | # to rearrange the headers. | 
|  | sed -e 's/^ *//' $_hdr | awk -v src=$_source ' | 
|  | BEGIN { | 
|  | date_seen=0 | 
|  | subject_seen=0 | 
|  | } | 
|  | /^commit/ { | 
|  | commit=$2 | 
|  | next; | 
|  | } | 
|  | /^Author:/ { | 
|  | split($0, a, ":") | 
|  | from=a[2] | 
|  | next; | 
|  | } | 
|  | /^Date:/ { date_seen=1; next } | 
|  | /^difflib/ { next } | 
|  |  | 
|  | // { | 
|  | if (date_seen == 0) | 
|  | next; | 
|  | if (subject_seen == 0) { | 
|  | if (length($0) != 0) { | 
|  | subject_seen=1 | 
|  | subject=$0; | 
|  | } | 
|  | next; | 
|  | } | 
|  | if (subject_seen == 1) { | 
|  | print subject | 
|  | print | 
|  | print "From:" from | 
|  | print | 
|  | print "Source " src " commit: " commit | 
|  | subject_seen=2 | 
|  | } | 
|  | print $0 | 
|  | }' > $_hdr.new | 
|  |  | 
|  | # Remove the last line if it contains only whitespace | 
|  | sed -i '${/^[[:space:]]*$/d;}' $_hdr.new | 
|  |  | 
|  | # Add Signed-off-by: header if specified | 
|  | if [ ! -z ${SIGNED_OFF_BY+x} ]; then | 
|  | add_header "Signed-off-by: $SIGNED_OFF_BY" $_hdr.new | 
|  | else	# get it from git config if present | 
|  | SOB_NAME=`git config --get user.name` | 
|  | SOB_EMAIL=`git config --get user.email` | 
|  | if [ ! -z ${SOB_NAME+x} ]; then | 
|  | add_header "Signed-off-by: $SOB_NAME <$SOB_EMAIL>" $_hdr.new | 
|  | fi | 
|  | fi | 
|  |  | 
|  | # now output the new patch | 
|  | cat $_hdr.new $_diff | 
|  |  | 
|  | rm -f $_hdr* $_diff | 
|  |  | 
|  | } | 
|  |  | 
|  | apply_patch() | 
|  | { | 
|  | local _patch=$1 | 
|  | local _patch_name=$2 | 
|  | local _current_commit=$3 | 
|  | local _new_patch=`mktemp` | 
|  | local _source="kernel" | 
|  | local _target="xfsprogs" | 
|  |  | 
|  | # filter just the libxfs parts of the patch | 
|  | if [ $SOURCE == "xfsprogs" ]; then | 
|  |  | 
|  | [ -n "$VERBOSE" ] || lsdiff $_patch | grep -q "[ab]/libxfs/" | 
|  | if [ $? -ne 0 ]; then | 
|  | echo "Doesn't look like an xfsprogs patch with libxfs changes" | 
|  | echo "Skipping commit $_current_commit" | 
|  | return | 
|  | fi | 
|  |  | 
|  | filter_kernel_patch $_patch > $_new_patch | 
|  | _source="xfsprogs" | 
|  | _target="kernel" | 
|  | elif [ $SOURCE == "kernel" ]; then | 
|  |  | 
|  | [ -n "$VERBOSE" ] || lsdiff $_patch | grep -q "[ab]/fs/xfs/libxfs/" | 
|  | if [ $? -ne 0 ]; then | 
|  | echo "Doesn't look like a kernel patch with libxfs changes" | 
|  | echo "Skipping commit $_current_commit" | 
|  | return | 
|  | fi | 
|  |  | 
|  | filter_xfsprogs_patch $_patch > $_new_patch | 
|  | else | 
|  | fail "Unknown source repo type: $SOURCE" | 
|  | fi | 
|  |  | 
|  | grep -q "Source $_target commit: " $_patch | 
|  | if [ "$?" -eq "0" ]; then | 
|  | echo "$_patch_name already synced up" | 
|  | echo "$_skipping commit $_current_commit" | 
|  | return | 
|  | fi | 
|  |  | 
|  | # now munge the header to be in the correct format. | 
|  | fixup_header_format $_source $_new_patch > $_new_patch.2 | 
|  |  | 
|  | if [ -n "$VERBOSE" ]; then | 
|  | echo "Filtered patch from $REPO contains:" | 
|  | lsdiff $_new_patch.2 | 
|  | fi | 
|  |  | 
|  | # Ok, now apply with guilt or patch; either may fail and require a force | 
|  | # and/or a manual reject fixup | 
|  | if [ $GUILT -eq 1 ]; then | 
|  | [ -n "$VERBOSE" ] || echo "$REPO looks like a guilt directory." | 
|  | PATCHES=`guilt applied | wc -l` | 
|  | if [ -n "$VERBOSE" -a $PATCHES -gt 0 ]; then | 
|  | echo -n "Top patch is: " | 
|  | guilt top | 
|  | fi | 
|  |  | 
|  | guilt import -P $_patch_name $_new_patch.2 | 
|  | guilt push | 
|  | if [ $? -eq 0 ]; then | 
|  | guilt refresh | 
|  | else | 
|  | echo "Guilt push of $_current_commit $_patch_name failed!" | 
|  | read -r -p "Skip or Fail [s|F]? " response | 
|  | if [ -z "$response" -o "$response" != "s" ]; then | 
|  | echo "Force push patch, fix and refresh." | 
|  | echo "Restart from commit $_current_commit" | 
|  | fail "Manual cleanup required!" | 
|  | else | 
|  | echo "Skipping." | 
|  | guilt delete -f $_patch_name | 
|  | fi | 
|  | fi | 
|  | elif [ $STGIT -eq 1 ]; then | 
|  | [ -n "$VERBOSE" ] || echo "$REPO looks like a stgit directory." | 
|  | PATCHES=`stg series | wc -l` | 
|  | if [ -n "$VERBOSE" -a $PATCHES -gt 0 ]; then | 
|  | echo -n "Top patch is: " | 
|  | stg top | 
|  | fi | 
|  |  | 
|  | stg import -n $_patch_name $_new_patch.2 | 
|  | if [ $? -ne 0 ]; then | 
|  | echo "stgit push failed!" | 
|  | read -r -p "Skip or Fail [s|F]? " response | 
|  | if [ -z "$response" -o "$response" != "s" ]; then | 
|  | echo "Force push patch, fix and refresh." | 
|  | echo "Restart from commit $_current_commit" | 
|  | fail "Manual cleanup required!" | 
|  | else | 
|  | echo "Skipping. Manual series file cleanup needed!" | 
|  | fi | 
|  | fi | 
|  | else | 
|  | echo "Applying with git am:" | 
|  | git am -s $_new_patch.2 | 
|  | echo "Patch was applied in $REPO; check for rejects, etc" | 
|  | fi | 
|  |  | 
|  | rm -f $_new_patch* | 
|  | } | 
|  |  | 
|  | # name a guilt patch. Code is lifted from guilt import-commit. | 
|  | name_patch() | 
|  | { | 
|  | s=`git log --no-decorate --pretty=oneline -1 $1 | cut -c 42-` | 
|  |  | 
|  | # Try to convert the first line of the commit message to a | 
|  | # valid patch name. | 
|  | fname=`printf %s "$s" |  \ | 
|  | sed -e "s/&/and/g" -e "s/[ :]/_/g" -e "s,[/\\],-,g" \ | 
|  | -e "s/['\\[{}]//g" -e 's/]//g' -e 's/\*/-/g' \ | 
|  | -e 's/\?/-/g' -e 's/\.\.\.*/./g' -e 's/^\.//' \ | 
|  | -e 's/\.patch$//' -e 's/\.$//' | tr A-Z a-z` | 
|  |  | 
|  | # Try harder to make it a legal commit name by | 
|  | # removing all but a few safe characters. | 
|  | fname=`echo $fname|tr -d -c _a-zA-Z0-9---/\\n` | 
|  |  | 
|  | echo $fname | 
|  | } | 
|  |  | 
|  | # single patch is easy. | 
|  | if [ -z "$COMMIT_ID" ]; then | 
|  | apply_patch $PATCH | 
|  | cleanup | 
|  | exit 0 | 
|  | fi | 
|  |  | 
|  | # switch to source repo and get individual commit IDs | 
|  | # | 
|  | # git rev-list gives us a list in reverse chronological order, so we need to | 
|  | # reverse that to give us the order we require. | 
|  | pushd $REPO > /dev/null | 
|  | check_repo Source | 
|  | hashr=`munge_hash_range $COMMIT_ID` | 
|  | if [ $SOURCE == "kernel" ]; then | 
|  | hashr="$hashr -- fs/xfs/libxfs" | 
|  | else | 
|  | hashr="$hashr -- libxfs" | 
|  | fi | 
|  |  | 
|  | # grab and echo the list of commits for confirmation | 
|  | echo "Commits to apply:" | 
|  | commit_list=`git rev-list $hashr | tac` | 
|  | git log --oneline $hashr |tac | 
|  | read -r -p "Proceed [y|N]? " response | 
|  | if [ -z "$response" -o "$response" != "y" ]; then | 
|  | fail "Aborted!" | 
|  | fi | 
|  | popd > /dev/null | 
|  |  | 
|  | PATCH=`mktemp` | 
|  | for commit in $commit_list; do | 
|  |  | 
|  | # switch to source repo and pull commit into a patch file | 
|  | pushd $REPO > /dev/null | 
|  | git show $commit > $PATCH || usage "Bad source commit ID!" | 
|  | patch_name=`name_patch $commit` | 
|  | popd > /dev/null | 
|  |  | 
|  | apply_patch $PATCH $patch_name $commit | 
|  | done | 
|  |  | 
|  |  | 
|  | cleanup |