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