Whamcloud - gitweb
back out previous commit, which was more extensive than i'd intended
[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         SYSIO_INTERFACE_DISPLAY_BLOCK;
65
66         SYSIO_INTERFACE_ENTER;
67
68         /*
69          * Neither old nor new may be the empty string.
70          */
71         if (*oldpath == '\0' || *newpath == '\0')
72                 SYSIO_INTERFACE_RETURN(-1, -ENOENT);
73
74         /*
75          * Resolve oldpath to a path node.
76          */
77         INTENT_INIT(&intent, INT_UPDPARENT, NULL, NULL);
78         err = _sysio_namei(_sysio_cwd, oldpath, ND_NOFOLLOW, &intent, &old);
79         if (err)
80                 goto error3;
81         /*
82          * Resolve newpath to a path node.
83          */
84         INTENT_INIT(&intent, INT_UPDPARENT, NULL, NULL);
85         err =
86             _sysio_namei(_sysio_cwd,
87                          newpath,
88                          ND_NOFOLLOW | ND_NEGOK,
89                          &intent,
90                          &new);
91         if (err)
92                 goto error2;
93
94         /*
95          * Don't allow mount points to move.
96          */
97         if (old->p_mount->mnt_root == old || old->p_cover ||
98             new->p_mount->mnt_root == new) {
99                 err = -EBUSY;
100                 goto error1;
101         }
102
103         /*
104          * No xdev renames either.
105          */
106         if (old->p_mount->mnt_fs != new->p_mount->mnt_fs) {
107                 err = -EXDEV;
108                 goto error1;
109         }
110
111         /*
112          * Make sure the old pnode can't be found in the ancestor chain
113          * for the new. If it can, they are trying to move into a subdirectory
114          * of the old.
115          */
116         nxtpb = new->p_base;
117         do {
118                 pb = nxtpb;
119                 nxtpb = pb->pb_parent;
120                 if (pb == old->p_base) {
121                         err = -EINVAL;
122                         goto error1;
123                 }
124         } while (nxtpb);
125
126         /*
127          * If old == new, we're done.
128          */
129         if (old->p_base->pb_ino == new->p_base->pb_ino)
130                 goto out;
131
132         if (new->p_base->pb_ino) {
133                 /*
134                  * Existing entry. We're replacing the new. Make sure that's
135                  * ok.
136                  */
137                 if (S_ISDIR(new->p_base->pb_ino->i_stbuf.st_mode)) {
138                         if (!S_ISDIR(old->p_base->pb_ino->i_stbuf.st_mode)) {
139                                 err = -EISDIR;
140                                 goto error1;
141                         }
142                         if (new->p_base->pb_ino->i_stbuf.st_nlink > 2) {
143                                 err = -ENOTEMPTY;
144                                 goto error1;
145                         }
146                 } else if (S_ISDIR(old->p_base->pb_ino->i_stbuf.st_mode)) {
147                         err = -ENOTDIR;
148                         goto error1;
149                 }
150         }
151
152         /*
153          * It's not impossible to clean up the altered name space after
154          * a rename. However, it is onerous and I don't want to do it right
155          * now. If it becomes an issue, we can do it later. For now, I've
156          * elected to use the semantic that says, basically, the entire
157          * sub-tree must be unreferenced. That's per POSIX, but it's a nasty
158          * thing to do to the caller.
159          */
160         if (_sysio_p_prune(new) != 1) {
161                 err = -EBUSY;
162                 goto error1;
163         }
164         /*
165          * Use the parent node operations to request the task in case the
166          * driver is implemented using differentiated inode operations based
167          * on file type, such as incore does.
168          */
169         err = old->p_parent->p_base->pb_ino->i_ops.inop_rename(old, new);
170         if (err)
171                 goto error1;
172         /*
173          * Reflect the successful rename in the active name space graph.
174          */
175         if (new->p_base->pb_ino)
176                 I_GONE(new->p_base->pb_ino);
177         new->p_base->pb_ino = old->p_base->pb_ino;
178         I_REF(new->p_base->pb_ino);
179
180 error1:
181         P_RELE(new);
182 error2:
183         P_RELE(old);
184 error3:
185         if (err)
186                 goto out;
187         _sysio_p_gone(old);                                     /* kill it! */
188 out:
189         SYSIO_INTERFACE_RETURN(err ? -1 : 0, err);
190 }