OLD | NEW |
1 #!/bin/sh | 1 #!/bin/sh |
2 # install - install a program, script, or datafile | 2 # install - install a program, script, or datafile |
3 | 3 |
4 scriptversion=2005-05-14.22 | 4 scriptversion=2006-12-25.00 |
5 | 5 |
6 # This originates from X11R5 (mit/util/scripts/install.sh), which was | 6 # This originates from X11R5 (mit/util/scripts/install.sh), which was |
7 # later released in X11R6 (xc/config/util/install.sh) with the | 7 # later released in X11R6 (xc/config/util/install.sh) with the |
8 # following copyright and license. | 8 # following copyright and license. |
9 # | 9 # |
10 # Copyright (C) 1994 X Consortium | 10 # Copyright (C) 1994 X Consortium |
11 # | 11 # |
12 # Permission is hereby granted, free of charge, to any person obtaining a copy | 12 # Permission is hereby granted, free of charge, to any person obtaining a copy |
13 # of this software and associated documentation files (the "Software"), to | 13 # of this software and associated documentation files (the "Software"), to |
14 # deal in the Software without restriction, including without limitation the | 14 # deal in the Software without restriction, including without limitation the |
(...skipping 17 matching lines...) Expand all Loading... |
32 # tium. | 32 # tium. |
33 # | 33 # |
34 # | 34 # |
35 # FSF changes to this file are in the public domain. | 35 # FSF changes to this file are in the public domain. |
36 # | 36 # |
37 # Calling this script install-sh is preferred over install.sh, to prevent | 37 # Calling this script install-sh is preferred over install.sh, to prevent |
38 # `make' implicit rules from creating a file called install from it | 38 # `make' implicit rules from creating a file called install from it |
39 # when there is no Makefile. | 39 # when there is no Makefile. |
40 # | 40 # |
41 # This script is compatible with the BSD install script, but was written | 41 # This script is compatible with the BSD install script, but was written |
42 # from scratch. It can only install one file at a time, a restriction | 42 # from scratch. |
43 # shared with many OS's install programs. | 43 |
| 44 nl=' |
| 45 ' |
| 46 IFS=" ""» $nl" |
44 | 47 |
45 # set DOITPROG to echo to test this script | 48 # set DOITPROG to echo to test this script |
46 | 49 |
47 # Don't use :- since 4.3BSD and earlier shells don't like it. | 50 # Don't use :- since 4.3BSD and earlier shells don't like it. |
48 doit="${DOITPROG-}" | 51 doit=${DOITPROG-} |
| 52 if test -z "$doit"; then |
| 53 doit_exec=exec |
| 54 else |
| 55 doit_exec=$doit |
| 56 fi |
49 | 57 |
50 # put in absolute paths if you don't have them in your path; or use env. vars. | 58 # Put in absolute file names if you don't have them in your path; |
| 59 # or use environment vars. |
51 | 60 |
52 mvprog="${MVPROG-mv}" | 61 chgrpprog=${CHGRPPROG-chgrp} |
53 cpprog="${CPPROG-cp}" | 62 chmodprog=${CHMODPROG-chmod} |
54 chmodprog="${CHMODPROG-chmod}" | 63 chownprog=${CHOWNPROG-chown} |
55 chownprog="${CHOWNPROG-chown}" | 64 cmpprog=${CMPPROG-cmp} |
56 chgrpprog="${CHGRPPROG-chgrp}" | 65 cpprog=${CPPROG-cp} |
57 stripprog="${STRIPPROG-strip}" | 66 mkdirprog=${MKDIRPROG-mkdir} |
58 rmprog="${RMPROG-rm}" | 67 mvprog=${MVPROG-mv} |
59 mkdirprog="${MKDIRPROG-mkdir}" | 68 rmprog=${RMPROG-rm} |
| 69 stripprog=${STRIPPROG-strip} |
60 | 70 |
61 chmodcmd="$chmodprog 0755" | 71 posix_glob='?' |
| 72 initialize_posix_glob=' |
| 73 test "$posix_glob" != "?" || { |
| 74 if (set -f) 2>/dev/null; then |
| 75 posix_glob= |
| 76 else |
| 77 posix_glob=: |
| 78 fi |
| 79 } |
| 80 ' |
| 81 |
| 82 posix_mkdir= |
| 83 |
| 84 # Desired mode of installed file. |
| 85 mode=0755 |
| 86 |
| 87 chgrpcmd= |
| 88 chmodcmd=$chmodprog |
62 chowncmd= | 89 chowncmd= |
63 chgrpcmd= | 90 mvcmd=$mvprog |
| 91 rmcmd="$rmprog -f" |
64 stripcmd= | 92 stripcmd= |
65 rmcmd="$rmprog -f" | 93 |
66 mvcmd="$mvprog" | |
67 src= | 94 src= |
68 dst= | 95 dst= |
69 dir_arg= | 96 dir_arg= |
70 dstarg= | 97 dst_arg= |
| 98 |
| 99 copy_on_change=false |
71 no_target_directory= | 100 no_target_directory= |
72 | 101 |
73 usage="Usage: $0 [OPTION]... [-T] SRCFILE DSTFILE | 102 usage="\ |
| 103 Usage: $0 [OPTION]... [-T] SRCFILE DSTFILE |
74 or: $0 [OPTION]... SRCFILES... DIRECTORY | 104 or: $0 [OPTION]... SRCFILES... DIRECTORY |
75 or: $0 [OPTION]... -t DIRECTORY SRCFILES... | 105 or: $0 [OPTION]... -t DIRECTORY SRCFILES... |
76 or: $0 [OPTION]... -d DIRECTORIES... | 106 or: $0 [OPTION]... -d DIRECTORIES... |
77 | 107 |
78 In the 1st form, copy SRCFILE to DSTFILE. | 108 In the 1st form, copy SRCFILE to DSTFILE. |
79 In the 2nd and 3rd, copy all SRCFILES to DIRECTORY. | 109 In the 2nd and 3rd, copy all SRCFILES to DIRECTORY. |
80 In the 4th, create DIRECTORIES. | 110 In the 4th, create DIRECTORIES. |
81 | 111 |
82 Options: | 112 Options: |
83 -c (ignored) | 113 --help display this help and exit. |
84 -d create directories instead of installing files. | 114 --version display version info and exit. |
85 -g GROUP $chgrpprog installed files to GROUP. | 115 |
86 -m MODE $chmodprog installed files to MODE. | 116 -c (ignored) |
87 -o USER $chownprog installed files to USER. | 117 -C install only if different (preserve the last data modification t
ime) |
88 -s $stripprog installed files. | 118 -d create directories instead of installing files. |
89 -t DIRECTORY install into DIRECTORY. | 119 -g GROUP $chgrpprog installed files to GROUP. |
90 -T report an error if DSTFILE is a directory. | 120 -m MODE $chmodprog installed files to MODE. |
91 --help display this help and exit. | 121 -o USER $chownprog installed files to USER. |
92 --version display version info and exit. | 122 -s $stripprog installed files. |
| 123 -t DIRECTORY install into DIRECTORY. |
| 124 -T report an error if DSTFILE is a directory. |
93 | 125 |
94 Environment variables override the default commands: | 126 Environment variables override the default commands: |
95 CHGRPPROG CHMODPROG CHOWNPROG CPPROG MKDIRPROG MVPROG RMPROG STRIPPROG | 127 CHGRPPROG CHMODPROG CHOWNPROG CMPPROG CPPROG MKDIRPROG MVPROG |
| 128 RMPROG STRIPPROG |
96 " | 129 " |
97 | 130 |
98 while test -n "$1"; do | 131 while test $# -ne 0; do |
99 case $1 in | 132 case $1 in |
100 -c) shift | 133 -c) ;; |
101 continue;; | |
102 | 134 |
103 -d) dir_arg=true | 135 -C) copy_on_change=true;; |
104 shift | 136 |
105 continue;; | 137 -d) dir_arg=true;; |
106 | 138 |
107 -g) chgrpcmd="$chgrpprog $2" | 139 -g) chgrpcmd="$chgrpprog $2" |
108 shift | 140 » shift;; |
109 shift | |
110 continue;; | |
111 | 141 |
112 --help) echo "$usage"; exit $?;; | 142 --help) echo "$usage"; exit $?;; |
113 | 143 |
114 -m) chmodcmd="$chmodprog $2" | 144 -m) mode=$2 |
115 shift | 145 » case $mode in |
116 shift | 146 » *' '* | *'» '* | *' |
117 continue;; | 147 '*» | *'*'* | *'?'* | *'['*) |
| 148 » echo "$0: invalid mode: $mode" >&2 |
| 149 » exit 1;; |
| 150 » esac |
| 151 » shift;; |
118 | 152 |
119 -o) chowncmd="$chownprog $2" | 153 -o) chowncmd="$chownprog $2" |
120 shift | 154 » shift;; |
121 shift | |
122 continue;; | |
123 | 155 |
124 -s) stripcmd=$stripprog | 156 -s) stripcmd=$stripprog;; |
125 shift | |
126 continue;; | |
127 | 157 |
128 -t) dstarg=$2 | 158 -t) dst_arg=$2 |
129 » shift | 159 » shift;; |
130 » shift | |
131 » continue;; | |
132 | 160 |
133 -T) no_target_directory=true | 161 -T) no_target_directory=true;; |
134 » shift | |
135 » continue;; | |
136 | 162 |
137 --version) echo "$0 $scriptversion"; exit $?;; | 163 --version) echo "$0 $scriptversion"; exit $?;; |
138 | 164 |
139 *) # When -d is used, all remaining arguments are directories to create. | 165 --)»shift |
140 » # When -t is used, the destination is already specified. | |
141 » test -n "$dir_arg$dstarg" && break | |
142 # Otherwise, the last argument is the destination. Remove it from $@. | |
143 » for arg | |
144 » do | |
145 if test -n "$dstarg"; then | |
146 » # $@ is not empty: it contains at least $arg. | |
147 » set fnord "$@" "$dstarg" | |
148 » shift # fnord | |
149 » fi | |
150 » shift # arg | |
151 » dstarg=$arg | |
152 » done | |
153 break;; | 166 break;; |
| 167 |
| 168 -*) echo "$0: invalid option: $1" >&2 |
| 169 exit 1;; |
| 170 |
| 171 *) break;; |
154 esac | 172 esac |
| 173 shift |
155 done | 174 done |
156 | 175 |
157 if test -z "$1"; then | 176 if test $# -ne 0 && test -z "$dir_arg$dst_arg"; then |
| 177 # When -d is used, all remaining arguments are directories to create. |
| 178 # When -t is used, the destination is already specified. |
| 179 # Otherwise, the last argument is the destination. Remove it from $@. |
| 180 for arg |
| 181 do |
| 182 if test -n "$dst_arg"; then |
| 183 # $@ is not empty: it contains at least $arg. |
| 184 set fnord "$@" "$dst_arg" |
| 185 shift # fnord |
| 186 fi |
| 187 shift # arg |
| 188 dst_arg=$arg |
| 189 done |
| 190 fi |
| 191 |
| 192 if test $# -eq 0; then |
158 if test -z "$dir_arg"; then | 193 if test -z "$dir_arg"; then |
159 echo "$0: no input file specified." >&2 | 194 echo "$0: no input file specified." >&2 |
160 exit 1 | 195 exit 1 |
161 fi | 196 fi |
162 # It's OK to call `install-sh -d' without argument. | 197 # It's OK to call `install-sh -d' without argument. |
163 # This can happen when creating conditional directories. | 198 # This can happen when creating conditional directories. |
164 exit 0 | 199 exit 0 |
165 fi | 200 fi |
166 | 201 |
| 202 if test -z "$dir_arg"; then |
| 203 trap '(exit $?); exit' 1 2 13 15 |
| 204 |
| 205 # Set umask so as not to create temps with too-generous modes. |
| 206 # However, 'strip' requires both read and write access to temps. |
| 207 case $mode in |
| 208 # Optimize common cases. |
| 209 *644) cp_umask=133;; |
| 210 *755) cp_umask=22;; |
| 211 |
| 212 *[0-7]) |
| 213 if test -z "$stripcmd"; then |
| 214 u_plus_rw= |
| 215 else |
| 216 u_plus_rw='% 200' |
| 217 fi |
| 218 cp_umask=`expr '(' 777 - $mode % 1000 ')' $u_plus_rw`;; |
| 219 *) |
| 220 if test -z "$stripcmd"; then |
| 221 u_plus_rw= |
| 222 else |
| 223 u_plus_rw=,u+rw |
| 224 fi |
| 225 cp_umask=$mode$u_plus_rw;; |
| 226 esac |
| 227 fi |
| 228 |
167 for src | 229 for src |
168 do | 230 do |
169 # Protect names starting with `-'. | 231 # Protect names starting with `-'. |
170 case $src in | 232 case $src in |
171 -*) src=./$src ;; | 233 -*) src=./$src;; |
172 esac | 234 esac |
173 | 235 |
174 if test -n "$dir_arg"; then | 236 if test -n "$dir_arg"; then |
175 dst=$src | 237 dst=$src |
176 src= | 238 dstdir=$dst |
| 239 test -d "$dstdir" |
| 240 dstdir_status=$? |
| 241 else |
177 | 242 |
178 if test -d "$dst"; then | |
179 mkdircmd=: | |
180 chmodcmd= | |
181 else | |
182 mkdircmd=$mkdirprog | |
183 fi | |
184 else | |
185 # Waiting for this to be detected by the "$cpprog $src $dsttmp" command | 243 # Waiting for this to be detected by the "$cpprog $src $dsttmp" command |
186 # might cause directories to be created, which would be especially bad | 244 # might cause directories to be created, which would be especially bad |
187 # if $src (and thus $dsttmp) contains '*'. | 245 # if $src (and thus $dsttmp) contains '*'. |
188 if test ! -f "$src" && test ! -d "$src"; then | 246 if test ! -f "$src" && test ! -d "$src"; then |
189 echo "$0: $src does not exist." >&2 | 247 echo "$0: $src does not exist." >&2 |
190 exit 1 | 248 exit 1 |
191 fi | 249 fi |
192 | 250 |
193 if test -z "$dstarg"; then | 251 if test -z "$dst_arg"; then |
194 echo "$0: no destination specified." >&2 | 252 echo "$0: no destination specified." >&2 |
195 exit 1 | 253 exit 1 |
196 fi | 254 fi |
197 | 255 |
198 dst=$dstarg | 256 dst=$dst_arg |
199 # Protect names starting with `-'. | 257 # Protect names starting with `-'. |
200 case $dst in | 258 case $dst in |
201 -*) dst=./$dst ;; | 259 -*) dst=./$dst;; |
202 esac | 260 esac |
203 | 261 |
204 # If destination is a directory, append the input filename; won't work | 262 # If destination is a directory, append the input filename; won't work |
205 # if double slashes aren't ignored. | 263 # if double slashes aren't ignored. |
206 if test -d "$dst"; then | 264 if test -d "$dst"; then |
207 if test -n "$no_target_directory"; then | 265 if test -n "$no_target_directory"; then |
208 » echo "$0: $dstarg: Is a directory" >&2 | 266 » echo "$0: $dst_arg: Is a directory" >&2 |
209 exit 1 | 267 exit 1 |
210 fi | 268 fi |
211 dst=$dst/`basename "$src"` | 269 dstdir=$dst |
| 270 dst=$dstdir/`basename "$src"` |
| 271 dstdir_status=0 |
| 272 else |
| 273 # Prefer dirname, but fall back on a substitute if dirname fails. |
| 274 dstdir=` |
| 275 » (dirname "$dst") 2>/dev/null || |
| 276 » expr X"$dst" : 'X\(.*[^/]\)//*[^/][^/]*/*$' \| \ |
| 277 » X"$dst" : 'X\(//\)[^/]' \| \ |
| 278 » X"$dst" : 'X\(//\)$' \| \ |
| 279 » X"$dst" : 'X\(/\)' \| . 2>/dev/null || |
| 280 » echo X"$dst" | |
| 281 » sed '/^X\(.*[^/]\)\/\/*[^/][^/]*\/*$/{ |
| 282 » » s//\1/ |
| 283 » » q |
| 284 » » } |
| 285 » » /^X\(\/\/\)[^/].*/{ |
| 286 » » s//\1/ |
| 287 » » q |
| 288 » » } |
| 289 » » /^X\(\/\/\)$/{ |
| 290 » » s//\1/ |
| 291 » » q |
| 292 » » } |
| 293 » » /^X\(\/\).*/{ |
| 294 » » s//\1/ |
| 295 » » q |
| 296 » » } |
| 297 » » s/.*/./; q' |
| 298 ` |
| 299 |
| 300 test -d "$dstdir" |
| 301 dstdir_status=$? |
212 fi | 302 fi |
213 fi | 303 fi |
214 | 304 |
215 # This sed command emulates the dirname command. | 305 obsolete_mkdir_used=false |
216 dstdir=`echo "$dst" | sed -e 's,/*$,,;s,[^/]*$,,;s,/*$,,;s,^$,.,'` | |
217 | 306 |
218 # Make sure that the destination directory exists. | 307 if test $dstdir_status != 0; then |
| 308 case $posix_mkdir in |
| 309 '') |
| 310 » # Create intermediate dirs using mode 755 as modified by the umask. |
| 311 » # This is like FreeBSD 'install' as of 1997-10-28. |
| 312 » umask=`umask` |
| 313 » case $stripcmd.$umask in |
| 314 » # Optimize common cases. |
| 315 » *[2367][2367]) mkdir_umask=$umask;; |
| 316 » .*0[02][02] | .[02][02] | .[02]) mkdir_umask=22;; |
219 | 317 |
220 # Skip lots of stat calls in the usual case. | 318 » *[0-7]) |
221 if test ! -d "$dstdir"; then | 319 » mkdir_umask=`expr $umask + 22 \ |
222 defaultIFS=' | 320 » - $umask % 100 % 40 + $umask % 20 \ |
223 » ' | 321 » - $umask % 10 % 4 + $umask % 2 |
224 IFS="${IFS-$defaultIFS}" | 322 » `;; |
| 323 » *) mkdir_umask=$umask,go-w;; |
| 324 » esac |
225 | 325 |
226 oIFS=$IFS | 326 » # With -d, create the new directory with the user-specified mode. |
227 # Some sh's can't handle IFS=/ for some reason. | 327 » # Otherwise, rely on $mkdir_umask. |
228 IFS='%' | 328 » if test -n "$dir_arg"; then |
229 set x `echo "$dstdir" | sed -e 's@/@%@g' -e 's@^%@/@'` | 329 » mkdir_mode=-m$mode |
230 shift | 330 » else |
231 IFS=$oIFS | 331 » mkdir_mode= |
| 332 » fi |
232 | 333 |
233 pathcomp= | 334 » posix_mkdir=false |
| 335 » case $umask in |
| 336 » *[123567][0-7][0-7]) |
| 337 » # POSIX mkdir -p sets u+wx bits regardless of umask, which |
| 338 » # is incompatible with FreeBSD 'install' when (umask & 300) != 0. |
| 339 » ;; |
| 340 » *) |
| 341 » tmpdir=${TMPDIR-/tmp}/ins$RANDOM-$$ |
| 342 » trap 'ret=$?; rmdir "$tmpdir/d" "$tmpdir" 2>/dev/null; exit $ret' 0 |
234 | 343 |
235 while test $# -ne 0 ; do | 344 » if (umask $mkdir_umask && |
236 pathcomp=$pathcomp$1 | 345 » » exec $mkdirprog $mkdir_mode -p -- "$tmpdir/d") >/dev/null 2>&1 |
| 346 » then |
| 347 » if test -z "$dir_arg" || { |
| 348 » » # Check for POSIX incompatibilities with -m. |
| 349 » » # HP-UX 11.23 and IRIX 6.5 mkdir -m -p sets group- or |
| 350 » » # other-writeable bit of parent directory when it shouldn't. |
| 351 » » # FreeBSD 6.1 mkdir -m -p sets mode of existing directory. |
| 352 » » ls_ld_tmpdir=`ls -ld "$tmpdir"` |
| 353 » » case $ls_ld_tmpdir in |
| 354 » » d????-?r-*) different_mode=700;; |
| 355 » » d????-?--*) different_mode=755;; |
| 356 » » *) false;; |
| 357 » » esac && |
| 358 » » $mkdirprog -m$different_mode -p -- "$tmpdir" && { |
| 359 » » ls_ld_tmpdir_1=`ls -ld "$tmpdir"` |
| 360 » » test "$ls_ld_tmpdir" = "$ls_ld_tmpdir_1" |
| 361 » » } |
| 362 » » } |
| 363 » then posix_mkdir=: |
| 364 » fi |
| 365 » rmdir "$tmpdir/d" "$tmpdir" |
| 366 » else |
| 367 » # Remove any dirs left behind by ancient mkdir implementations. |
| 368 » rmdir ./$mkdir_mode ./-p ./-- 2>/dev/null |
| 369 » fi |
| 370 » trap '' 0;; |
| 371 » esac;; |
| 372 esac |
| 373 |
| 374 if |
| 375 $posix_mkdir && ( |
| 376 » umask $mkdir_umask && |
| 377 » $doit_exec $mkdirprog $mkdir_mode -p -- "$dstdir" |
| 378 ) |
| 379 then : |
| 380 else |
| 381 |
| 382 # The umask is ridiculous, or mkdir does not conform to POSIX, |
| 383 # or it failed possibly due to a race condition. Create the |
| 384 # directory the slow way, step by step, checking for races as we go. |
| 385 |
| 386 case $dstdir in |
| 387 » /*) prefix='/';; |
| 388 » -*) prefix='./';; |
| 389 » *) prefix='';; |
| 390 esac |
| 391 |
| 392 eval "$initialize_posix_glob" |
| 393 |
| 394 oIFS=$IFS |
| 395 IFS=/ |
| 396 $posix_glob set -f |
| 397 set fnord $dstdir |
237 shift | 398 shift |
238 if test ! -d "$pathcomp"; then | 399 $posix_glob set +f |
239 $mkdirprog "$pathcomp" | 400 IFS=$oIFS |
240 » # mkdir can fail with a `File exist' error in case several | 401 |
241 » # install-sh are creating the directory concurrently. This | 402 prefixes= |
242 » # is OK. | 403 |
243 » test -d "$pathcomp" || exit | 404 for d |
| 405 do |
| 406 » test -z "$d" && continue |
| 407 |
| 408 » prefix=$prefix$d |
| 409 » if test -d "$prefix"; then |
| 410 » prefixes= |
| 411 » else |
| 412 » if $posix_mkdir; then |
| 413 » (umask=$mkdir_umask && |
| 414 » $doit_exec $mkdirprog $mkdir_mode -p -- "$dstdir") && break |
| 415 » # Don't fail if two instances are running concurrently. |
| 416 » test -d "$prefix" || exit 1 |
| 417 » else |
| 418 » case $prefix in |
| 419 » *\'*) qprefix=`echo "$prefix" | sed "s/'/'\\\\\\\\''/g"`;; |
| 420 » *) qprefix=$prefix;; |
| 421 » esac |
| 422 » prefixes="$prefixes '$qprefix'" |
| 423 » fi |
| 424 » fi |
| 425 » prefix=$prefix/ |
| 426 done |
| 427 |
| 428 if test -n "$prefixes"; then |
| 429 » # Don't fail if two instances are running concurrently. |
| 430 » (umask $mkdir_umask && |
| 431 » eval "\$doit_exec \$mkdirprog $prefixes") || |
| 432 » test -d "$dstdir" || exit 1 |
| 433 » obsolete_mkdir_used=true |
244 fi | 434 fi |
245 pathcomp=$pathcomp/ | 435 fi |
246 done | |
247 fi | 436 fi |
248 | 437 |
249 if test -n "$dir_arg"; then | 438 if test -n "$dir_arg"; then |
250 $doit $mkdircmd "$dst" \ | 439 { test -z "$chowncmd" || $doit $chowncmd "$dst"; } && |
251 && { test -z "$chowncmd" || $doit $chowncmd "$dst"; } \ | 440 { test -z "$chgrpcmd" || $doit $chgrpcmd "$dst"; } && |
252 && { test -z "$chgrpcmd" || $doit $chgrpcmd "$dst"; } \ | 441 { test "$obsolete_mkdir_used$chowncmd$chgrpcmd" = false || |
253 && { test -z "$stripcmd" || $doit $stripcmd "$dst"; } \ | 442 test -z "$chmodcmd" || $doit $chmodcmd $mode "$dst"; } || exit 1 |
254 && { test -z "$chmodcmd" || $doit $chmodcmd "$dst"; } | |
255 | |
256 else | 443 else |
257 dstfile=`basename "$dst"` | |
258 | 444 |
259 # Make a couple of temp file names in the proper directory. | 445 # Make a couple of temp file names in the proper directory. |
260 dsttmp=$dstdir/_inst.$$_ | 446 dsttmp=$dstdir/_inst.$$_ |
261 rmtmp=$dstdir/_rm.$$_ | 447 rmtmp=$dstdir/_rm.$$_ |
262 | 448 |
263 # Trap to clean up those temp files at exit. | 449 # Trap to clean up those temp files at exit. |
264 trap 'ret=$?; rm -f "$dsttmp" "$rmtmp" && exit $ret' 0 | 450 trap 'ret=$?; rm -f "$dsttmp" "$rmtmp" && exit $ret' 0 |
265 trap '(exit $?); exit' 1 2 13 15 | |
266 | 451 |
267 # Copy the file name to the temp name. | 452 # Copy the file name to the temp name. |
268 $doit $cpprog "$src" "$dsttmp" && | 453 (umask $cp_umask && $doit_exec $cpprog "$src" "$dsttmp") && |
269 | 454 |
270 # and set any options; do chmod last to preserve setuid bits. | 455 # and set any options; do chmod last to preserve setuid bits. |
271 # | 456 # |
272 # If any of these fail, we abort the whole thing. If we want to | 457 # If any of these fail, we abort the whole thing. If we want to |
273 # ignore errors from any of these, just make sure not to ignore | 458 # ignore errors from any of these, just make sure not to ignore |
274 # errors from the above "$doit $cpprog $src $dsttmp" command. | 459 # errors from the above "$doit $cpprog $src $dsttmp" command. |
275 # | 460 # |
276 { test -z "$chowncmd" || $doit $chowncmd "$dsttmp"; } \ | 461 { test -z "$chowncmd" || $doit $chowncmd "$dsttmp"; } && |
277 && { test -z "$chgrpcmd" || $doit $chgrpcmd "$dsttmp"; } \ | 462 { test -z "$chgrpcmd" || $doit $chgrpcmd "$dsttmp"; } && |
278 && { test -z "$stripcmd" || $doit $stripcmd "$dsttmp"; } \ | 463 { test -z "$stripcmd" || $doit $stripcmd "$dsttmp"; } && |
279 && { test -z "$chmodcmd" || $doit $chmodcmd "$dsttmp"; } && | 464 { test -z "$chmodcmd" || $doit $chmodcmd $mode "$dsttmp"; } && |
280 | 465 |
281 # Now rename the file to the real destination. | 466 # If -C, don't bother to copy if it wouldn't change the file. |
282 { $doit $mvcmd -f "$dsttmp" "$dstdir/$dstfile" 2>/dev/null \ | 467 if $copy_on_change && |
283 || { | 468 old=`LC_ALL=C ls -dlL "$dst"» 2>/dev/null` && |
284 » # The rename failed, perhaps because mv can't rename something else | 469 new=`LC_ALL=C ls -dlL "$dsttmp"» 2>/dev/null` && |
285 » # to itself, or perhaps because mv is so ancient that it does not | |
286 » # support -f. | |
287 | 470 |
288 » # Now remove or move aside any old file at destination location. | 471 eval "$initialize_posix_glob" && |
289 » # We try this two ways since rm can't unlink itself on some | 472 $posix_glob set -f && |
290 » # systems and the destination file might be busy for other | 473 set X $old && old=:$2:$4:$5:$6 && |
291 » # reasons. In this case, the final cleanup might fail but the new | 474 set X $new && new=:$2:$4:$5:$6 && |
292 » # file should still install successfully. | 475 $posix_glob set +f && |
293 » { | |
294 » if test -f "$dstdir/$dstfile"; then | |
295 » $doit $rmcmd -f "$dstdir/$dstfile" 2>/dev/null \ | |
296 » || $doit $mvcmd -f "$dstdir/$dstfile" "$rmtmp" 2>/dev/null \ | |
297 » || { | |
298 » » echo "$0: cannot unlink or rename $dstdir/$dstfile" >&2 | |
299 » » (exit 1); exit 1 | |
300 » } | |
301 » else | |
302 » : | |
303 » fi | |
304 » } && | |
305 | 476 |
306 » # Now rename the file to the real destination. | 477 test "$old" = "$new" && |
307 » $doit $mvcmd "$dsttmp" "$dstdir/$dstfile" | 478 $cmpprog "$dst" "$dsttmp" >/dev/null 2>&1 |
308 » } | 479 then |
309 } | 480 rm -f "$dsttmp" |
310 fi || { (exit 1); exit 1; } | 481 else |
| 482 # Rename the file to the real destination. |
| 483 $doit $mvcmd -f "$dsttmp" "$dst" 2>/dev/null || |
| 484 |
| 485 # The rename failed, perhaps because mv can't rename something else |
| 486 # to itself, or perhaps because mv is so ancient that it does not |
| 487 # support -f. |
| 488 { |
| 489 » # Now remove or move aside any old file at destination location. |
| 490 » # We try this two ways since rm can't unlink itself on some |
| 491 » # systems and the destination file might be busy for other |
| 492 » # reasons. In this case, the final cleanup might fail but the new |
| 493 » # file should still install successfully. |
| 494 » { |
| 495 » test ! -f "$dst" || |
| 496 » $doit $rmcmd -f "$dst" 2>/dev/null || |
| 497 » { $doit $mvcmd -f "$dst" "$rmtmp" 2>/dev/null && |
| 498 » { $doit $rmcmd -f "$rmtmp" 2>/dev/null; :; } |
| 499 » } || |
| 500 » { echo "$0: cannot unlink or rename $dst" >&2 |
| 501 » (exit 1); exit 1 |
| 502 » } |
| 503 » } && |
| 504 |
| 505 » # Now rename the file to the real destination. |
| 506 » $doit $mvcmd "$dsttmp" "$dst" |
| 507 } |
| 508 fi || exit 1 |
| 509 |
| 510 trap '' 0 |
| 511 fi |
311 done | 512 done |
312 | 513 |
313 # The final little trick to "correctly" pass the exit status to the exit trap. | |
314 { | |
315 (exit 0); exit 0 | |
316 } | |
317 | |
318 # Local variables: | 514 # Local variables: |
319 # eval: (add-hook 'write-file-hooks 'time-stamp) | 515 # eval: (add-hook 'write-file-hooks 'time-stamp) |
320 # time-stamp-start: "scriptversion=" | 516 # time-stamp-start: "scriptversion=" |
321 # time-stamp-format: "%:y-%02m-%02d.%02H" | 517 # time-stamp-format: "%:y-%02m-%02d.%02H" |
322 # time-stamp-end: "$" | 518 # time-stamp-end: "$" |
323 # End: | 519 # End: |
OLD | NEW |