WARNS6 clean.
[dragonfly.git] / usr.bin / ar / move.c
1 /*-
2  * Copyright (c) 1990, 1993, 1994
3  *      The Regents of the University of California.  All rights reserved.
4  *
5  * This code is derived from software contributed to Berkeley by
6  * Hugh Smith at The University of Guelph.
7  *
8  * Redistribution and use in source and binary forms, with or without
9  * modification, are permitted provided that the following conditions
10  * are met:
11  * 1. Redistributions of source code must retain the above copyright
12  *    notice, this list of conditions and the following disclaimer.
13  * 2. Redistributions in binary form must reproduce the above copyright
14  *    notice, this list of conditions and the following disclaimer in the
15  *    documentation and/or other materials provided with the distribution.
16  * 3. All advertising materials mentioning features or use of this software
17  *    must display the following acknowledgement:
18  *      This product includes software developed by the University of
19  *      California, Berkeley and its contributors.
20  * 4. Neither the name of the University nor the names of its contributors
21  *    may be used to endorse or promote products derived from this software
22  *    without specific prior written permission.
23  *
24  * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND
25  * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
26  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
27  * ARE DISCLAIMED.  IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE
28  * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
29  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
30  * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
31  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
32  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
33  * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
34  * SUCH DAMAGE.
35  *
36  * @(#)move.c   8.3 (Berkeley) 4/2/94
37  *
38  * $DragonFly: src/usr.bin/ar/Attic/move.c,v 1.4 2005/01/13 18:57:56 okumoto Exp $
39  */
40
41 #include <sys/param.h>
42 #include <sys/stat.h>
43
44 #include <ar.h>
45 #include <dirent.h>
46 #include <err.h>
47 #include <fcntl.h>
48 #include <stdio.h>
49 #include <unistd.h>
50
51 #include "archive.h"
52 #include "extern.h"
53 #include "pathnames.h"
54
55 /*
56  * move --
57  *      Change location of named members in archive - if 'b' or 'i' option
58  *      selected then named members are placed before 'posname'.  If 'a'
59  *      option selected members go after 'posname'.  If no options, members
60  *      are moved to end of archive.
61  */
62 int
63 move(char **argv)
64 {
65         CF cf;
66         off_t size, tsize;
67         int afd, curfd, mods, tfd1, tfd2, tfd3;
68         char *file;
69
70         afd = open_archive(O_RDWR);
71         mods = options & (AR_A|AR_B);
72
73         tfd1 = tmp();                   /* Files before key file. */
74         tfd2 = tmp();                   /* Files selected by user. */
75         tfd3 = tmp();                   /* Files after key file. */
76
77         /*
78          * Break archive into three parts -- selected entries and entries
79          * before and after the key entry.  If positioning before the key,
80          * place the key at the beginning of the after key entries and if
81          * positioning after the key, place the key at the end of the before
82          * key entries.  Put it all back together at the end.
83          */
84
85         /* Read and write to an archive; pad on both. */
86         SETCF(afd, archive, 0, tname, RPAD|WPAD);
87         for (curfd = tfd1; get_arobj(afd);) {
88                 if (*argv && (file = files(argv))) {
89                         if (options & AR_V)
90                                 printf("m - %s\n", file);
91                         cf.wfd = tfd2;
92                         put_arobj(&cf, (struct stat *)NULL);
93                         continue;
94                 }
95                 if (mods && compare(posname)) {
96                         mods = 0;
97                         if (options & AR_B)
98                                 curfd = tfd3;
99                         cf.wfd = curfd;
100                         put_arobj(&cf, (struct stat *)NULL);
101                         if (options & AR_A)
102                                 curfd = tfd3;
103                 } else {
104                         cf.wfd = curfd;
105                         put_arobj(&cf, (struct stat *)NULL);
106                 }
107         }
108
109         if (mods) {
110                 warnx("%s: archive member not found", posarg);
111                 close_archive(afd);
112                 return (1);
113         }
114         lseek(afd, (off_t)SARMAG, SEEK_SET);
115
116         SETCF(tfd1, tname, afd, archive, NOPAD);
117         tsize = size = lseek(tfd1, (off_t)0, SEEK_CUR);
118         lseek(tfd1, (off_t)0, SEEK_SET);
119         copy_ar(&cf, size);
120
121         tsize += size = lseek(tfd2, (off_t)0, SEEK_CUR);
122         lseek(tfd2, (off_t)0, SEEK_SET);
123         cf.rfd = tfd2;
124         copy_ar(&cf, size);
125
126         tsize += size = lseek(tfd3, (off_t)0, SEEK_CUR);
127         lseek(tfd3, (off_t)0, SEEK_SET);
128         cf.rfd = tfd3;
129         copy_ar(&cf, size);
130
131         ftruncate(afd, tsize + SARMAG);
132         close_archive(afd);
133
134         if (*argv) {
135                 orphans(argv);
136                 return (1);
137         }
138         return (0);
139 }