Whamcloud - gitweb
new upstream libsysio snapshot (20041101)
[fs/lustre-release.git] / libsysio / src / link.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 <string.h>
45 #include <errno.h>
46 #include <assert.h>
47 #include <sys/types.h>
48 #include <sys/stat.h>
49 #include <unistd.h>
50 #include <sys/queue.h>
51
52 #include "sysio.h"
53 #include "mount.h"
54 #include "inode.h"
55 #include "sysio-symbols.h"
56
57 int
58 SYSIO_INTERFACE_NAME(link)(const char *oldpath, const char *newpath)
59 {
60         struct intent intent;
61         int     err;
62         struct pnode *old, *new;
63         SYSIO_INTERFACE_DISPLAY_BLOCK;
64
65         SYSIO_INTERFACE_ENTER;
66         INTENT_INIT(&intent, 0, NULL, NULL);
67         err = _sysio_namei(_sysio_cwd, oldpath, 0, &intent, &old);
68         if (err)
69                 goto out;
70         if (S_ISDIR(old->p_base->pb_ino->i_stbuf.st_mode)) {
71                 err = -EPERM;
72                 goto error1;
73         }
74         INTENT_INIT(&intent, INT_UPDPARENT, NULL, NULL);
75         new = NULL;
76         err = _sysio_namei(_sysio_cwd, newpath, ND_NEGOK, &intent, &new);
77         if (err)
78                 goto error1;
79         if (new->p_base->pb_ino) {
80                 err = -EEXIST;
81                 goto error2;
82         }
83         if (old->p_mount->mnt_root != new->p_mount->mnt_root) {
84                 err = -EXDEV;
85                 goto error2;
86         }
87         err = old->p_base->pb_ino->i_ops.inop_link(old, new);
88         if (err)
89                 goto error2;
90         /*
91          * The new p-node must be pointed at the inode referenced by the old.
92          */
93         assert(!new->p_base->pb_ino && old->p_base->pb_ino);
94         new->p_base->pb_ino = old->p_base->pb_ino;
95         I_REF(new->p_base->pb_ino);
96
97 error2:
98         P_RELE(new);
99 error1:
100         P_RELE(old);
101 out:
102         SYSIO_INTERFACE_RETURN(err ? -1 : 0, err);
103 }
104
105 #ifdef REDSTORM
106 #undef __link
107 sysio_sym_weak_alias(SYSIO_INTERFACE_NAME(link), 
108                      PREPEND(__, SYSIO_INTERFACE_NAME(link)))
109 #endif