1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
|
#!/bin/bash
# Create a pristine environment to minimize unnecessary fuzz when different
# users use git-ptx-patches on the same patch stack. That is, don't load any
# config files, and pin down environment variables which could influence git's
# behaviour or patch output.
pristine_git() {
# Notes from the git(1) manpage:
# - GIT_DIFF_OPTS takes takes precedence over -U command line parameter
HOME=/nonexistent \
XDG_CONFIG_HOME=/nonexistent \
GIT_CONFIG_NOSYSTEM=true \
GIT_DIFF_OPTS="-u3" \
git "$@"
}
GIT="pristine_git"
PTX_PATCHES_HEADER="# generated by git-ptx-patches"
function _md5sum() {
local sum=$(md5sum)
echo "# $sum git-ptx-patches magic"
}
if [ ! -L .ptxdist/patches ]; then
echo "Error: This is not patched by ptxdist. Aborting."
exit 1
fi
if [ ! -L .ptxdist/series ]; then
echo "Error: .ptxdist/series must be a symbolic link. Aborting."
exit 1
fi
remove_old=no
tag=base
numbered_patches=true
all_tags=false
if grep -q "$PTX_PATCHES_HEADER" .ptxdist/series; then
echo "Found series file generated by git-ptx-patches."
lines=$(wc -l < .ptxdist/series)
lines=$[lines-1]
magic=$(head -n$lines .ptxdist/series | _md5sum)
if grep -q "^$magic" .ptxdist/series; then
remove_old=yes
else
echo "Warning: .ptxdist/series was modified."
fi
fi
if [ "x$1" = "x--force-remove" ]; then
remove_old="force"
shift
fi
while getopts "aft:n" opt; do
case "${opt}" in
a)
all_tags=true
;;
f)
remove_old="force"
;;
t)
tag="${OPTARG}"
;;
n)
numbered_patches=false
;;
esac
done
shift $((${OPTIND} - 1))
tag2=$(grep "#tag:" .ptxdist/series | awk "/#tag:${tag} /{if (getline) print \$1}" | sed 's,#tag:,,')
range="${tag}..${tag2}"
if "${all_tags}" ; then
args=()
if [ "${remove_old}" = "force" ]; then
args[${#args[*]}]="-f"
fi
if ! "${numbered_patches}"; then
args[${#args[*]}]="-n"
fi
for tag in $(sed -n 's/^#tag:\([^ ]*\) .*/\1/p' .ptxdist/series .ptxdist/series.append); do
echo "Updating patches for '${tag}'..."
"$0" "${args[@]}" -t "${tag}" || break
echo
done
exit
fi
echo "$PTX_PATCHES_HEADER" > .ptxdist/series.0
:> .ptxdist/series.1
touch .ptxdist/series.append
if grep -q "^#tag:" .ptxdist/series .ptxdist/series.append; then
tagline=$(cat .ptxdist/series .ptxdist/series.append | grep "#tag:${tag}")
t=$(echo "${tagline}"|cut -d' ' -f1)
if [ "#tag:${tag}" == "${t}" ]; then
tagopt=$(echo "${tagline}"|cut -d' ' -s -f2-)
sed -e "/$PTX_PATCHES_HEADER/d" -n \
-e '/git-ptx-patches magic/d' \
-e "0,/#tag:${tag}/p" \
.ptxdist/series .ptxdist/series.append >> .ptxdist/series.0
# Remove patches before #tag:${tag} so they don't get rm'd with remove_old=yes
sed -i --follow-symlinks "0,/#tag:${tag}/d" .ptxdist/series
if [ -n "${tag2}" ]; then
sed -n -e "/#tag:${tag2}/,/git-ptx-patches magic/p" .ptxdist/series > .ptxdist/series.1
sed -i "/git-ptx-patches magic/d" .ptxdist/series.1
sed -i --follow-symlinks "/#tag:${tag2}/,/git-ptx-patches magic/d" .ptxdist/series
fi
else
echo "series contains #tag:* lines, but could not find #tag:${tag} line in series. Aborting."
exit 1
fi
else
if [ "${tag}" != "base" ]; then
echo "When using series with no #tag:* lines, you must use base tag."
exit 1
fi
echo "#tag:${tag} --start-number 1" >> .ptxdist/series.0
fi
rm .ptxdist/series.append
case "$remove_old" in
"no") ;;
"yes")
echo "Removing old patches ..."
while read patch para; do
case "${patch}" in
""|"#"*) continue ;;
*) rm .ptxdist/patches/$patch ;;
esac
done < .ptxdist/series
;;
"force")
echo "Removing old patches (forced) ..."
find .ptxdist/patches/ | while read file; do
case "$file" in
".ptxdist/patches/") continue ;;
".ptxdist/patches/series") continue ;;
".ptxdist/patches/autogen.sh") continue ;;
*)
if grep -q "${file##\.ptxdist/patches/}" .ptxdist/series.{0,1}; then
echo "Keep base patch ${file}"
else
rm -rf "$file"
fi
;;
esac
done
;;
esac
# git-format-patch --no-signature is supported since git 1.7.2
if ${GIT} format-patch -h 2>&1 | grep -q signature; then
GIT_EXTRA_ARGS="--no-signature"
fi
# git-format-patch --notes is supported since git 1.7.6, but actually you want
# git 1.8.1-rc0 to get the notes below the --- marker
if man git-format-patch | grep -e --notes > /dev/null; then
GIT_EXTRA_ARGS="$GIT_EXTRA_ARGS --notes"
fi
GIT_EXTRA_ARGS="$GIT_EXTRA_ARGS --summary --stat=80"
cat .ptxdist/series.0 > .ptxdist/series
${GIT} format-patch -N $GIT_EXTRA_ARGS ${tagopt} -o .ptxdist/patches/ ${range} | while read patch; do
if "$numbered_patches"; then
patchname="${patch#.ptxdist/patches/}"
else
patchname="${patch#.ptxdist/patches/[0-9][0-9][0-9][0-9]-}"
mv -n "$patch" ".ptxdist/patches/$patchname"
fi
echo "$patchname"
done > .ptxdist/series.auto
cat .ptxdist/series.auto >> .ptxdist/series
cat .ptxdist/series.1 >> .ptxdist/series
cat .ptxdist/series | _md5sum >> .ptxdist/series
# The first line of the patch is 'From <some-git-hash> ...'
# remove it to avoid unnecessary changes in the patch files.
find .ptxdist/patches/ ! -type d | sed -e 's,^.ptxdist/patches/,,' | \
while read patch para; do
case "$patch" in
"series"|"autogen.sh") continue ;;
*) ;;
esac
if grep -q "$patch" .ptxdist/series.auto; then
p=".ptxdist/patches/$patch"
tail -n+2 "$p" > ".$patch.ptx-patches"
mv ".$patch.ptx-patches" "$p"
else
if grep -q "$patch" .ptxdist/series.{0,1}; then
echo "Base patch \"$patch\"!"
else
echo "Old patch \"$patch\"!"
fi
fi
done | sort
|