Whamcloud - gitweb
import older libsysio snapshot.
[fs/lustre-release.git] / libsysio / src / rename.c
1 /*
2  *    This Cplant(TM) source code is the property of Sandia National
3  *    Laboratories.
4  *
5  *    This Cplant(TM) source code is copyrighted by Sandia National
6  *    Laboratories.
7  *
8  *    The redistribution of this Cplant(TM) source code is subject to the
9  *    terms of the GNU Lesser General Public License
10  *    (see cit/LGPL or http://www.gnu.org/licenses/lgpl.html)
11  *
12  *    Cplant(TM) Copyright 1998-2003 Sandia Corporation. 
13  *    Under the terms of Contract DE-AC04-94AL85000, there is a non-exclusive
14  *    license for use of this work by or on behalf of the US Government.
15  *    Export of this program may require a license from the United States
16  *    Government.
17  */
18
19 /*
20  * This library is free software; you can redistribute it and/or
21  * modify it under the terms of the GNU Lesser General Public
22  * License as published by the Free Software Foundation; either
23  * version 2.1 of the License, or (at your option) any later version.
24  * 
25  * This library is distributed in the hope that it will be useful,
26  * but WITHOUT ANY WARRANTY; without even the implied warranty of
27  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
28  * Lesser General Public License for more details.
29  * 
30  * You should have received a copy of the GNU Lesser General Public
31  * License along with this library; if not, write to the Free Software
32  * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
33  *
34  * Questions or comments about this library should be sent to:
35  *
36  * Lee Ward
37  * Sandia National Laboratories, New Mexico
38  * P.O. Box 5800
39  * Albuquerque, NM 87185-1110
40  *
41  * lee@sandia.gov
42  */
43
44 #include <stdlib.h>
45 #include <string.h>
46 #include <errno.h>
47 #include <assert.h>
48 #include <sys/types.h>
49 #include <sys/stat.h>
50 #include <unistd.h>
51 #include <sys/queue.h>
52
53 #include "sysio.h"
54 #include "mount.h"
55 #include "inode.h"
56
57 int
58 SYSIO_INTERFACE_NAME(rename)(const char *oldpath, const char *newpath)
59 {
60         struct intent intent;
61         int     err;
62         struct pnode *old, *new;
63         struct pnode_base *nxtpb, *pb;
64         struct intnl_stat ostbuf, nstbuf;
65         SYSIO_INTERFACE_DISPLAY_BLOCK;
66
67         SYSIO_INTERFACE_ENTER;
68         /*
69          * Resolve oldpath to a path node.
70          */
71         INTENT_INIT(&intent, INT_UPDPARENT, NULL, NULL);
72         err = _sysio_namei(_sysio_cwd, oldpath, 0, &intent, &old);
73         if (err)
74                 goto error3;
75         /*
76          * Resolve newpath to a path node.
77          */
78         INTENT_INIT(&intent, INT_UPDPARENT, NULL, NULL);
79         err = _sysio_namei(_sysio_cwd, newpath, ND_NEGOK, &intent, &new);
80         if (err && !new)
81                 goto error2;
82
83         if (old->p_mount->mnt_root == old || old->p_cover ||
84             new->p_mount->mnt_root == new) {
85                 err = -EBUSY;
86                 goto error1;
87         }
88
89         if (old->p_mount->mnt_fs != new->p_mount->mnt_fs) {
90                 /*
91                  * Oops. They're trying to move it across file systems.
92                  */
93                 err = -EXDEV;
94                 goto error1;
95         }
96
97         /*
98          * Make sure the old pnode can't be found in the ancestor chain
99          * for the new. If it can, they are trying to move into a subdirectory
100          * of the old.
101          */
102         nxtpb = new->p_base;
103         do {
104                 pb = nxtpb;
105                 nxtpb = pb->pb_parent;
106                 if (pb == old->p_base) {
107                         err = -EINVAL;
108                         goto error1;
109                 }
110         } while (nxtpb);
111
112         while (new->p_base->pb_ino) {
113                 /*
114                  * Existing entry. We're replacing the new. Make sure that's
115                  * ok.
116                  */
117                 err =
118                     old->p_base->pb_ino->i_ops.inop_getattr(old, NULL, &ostbuf);
119                 if (err)
120                         goto error1;
121                 err =
122                     new->p_base->pb_ino->i_ops.inop_getattr(new, NULL, &nstbuf);
123                 if (err) {
124                         if (err != ENOENT)
125                                 goto error1;
126                         /*
127                          * Rats! It disappeared beneath us.
128                          */
129                         (void )_sysio_p_validate(new, NULL, NULL);
130                         continue;
131                 }
132                 if (S_ISDIR(ostbuf.st_mode)) {
133                         if (!S_ISDIR(nstbuf.st_mode)) {
134                                 err = -ENOTDIR;
135                                 goto error1;
136                         }
137                         if (nstbuf.st_nlink > 2) {
138                                 err = -ENOTEMPTY;
139                                 goto error1;
140                         }
141                 } else if (S_ISDIR(nstbuf.st_mode)) {
142                         err = -EEXIST;
143                         goto error1;
144                 }
145                 break;
146         }
147
148         /*
149          * It's not impossible to clean up the altered name space after
150          * a rename. However, it is onerous and I don't want to do it right
151          * now. If it becomes an issue, we can do it later. For now, I've
152          * elected to use the semantic that says, basically, the entire
153          * sub-tree must be unreferenced. That's per POSIX, but it's a nasty
154          * this to do to the caller.
155          */
156         if (_sysio_p_prune(new) != 1) {
157                 err = -EBUSY;
158                 goto error1;
159         }
160         err = old->p_base->pb_ino->i_ops.inop_rename(old, new);
161         if (err)
162                 goto error1;
163         /*
164          * Reflect the successful rename in the active name space graph.
165          */
166         if (new->p_base->pb_ino)
167                 I_GONE(new->p_base->pb_ino);
168         new->p_base->pb_ino = old->p_base->pb_ino;
169         I_REF(new->p_base->pb_ino);
170
171 error1:
172         P_RELE(new);
173 error2:
174         P_RELE(old);
175 error3:
176         if (err)
177                 goto out;
178         _sysio_p_gone(old);                                     /* kill it! */
179 out:
180         SYSIO_INTERFACE_RETURN(err ? -1 : 0, err);
181 }