Whamcloud - gitweb
LU-5456 hsm: hold inode mutex around ll_setattr_raw()
[fs/lustre-release.git] / libsysio / src / mknod.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 <unistd.h>
45 #include <errno.h>
46 #include <assert.h>
47 #include <sys/types.h>
48 #include <sys/stat.h>
49 #include <sys/queue.h>
50
51 #include "sysio.h"
52 #include "inode.h"
53 #include "fs.h"
54 #include "mount.h"
55
56 #include "sysio-symbols.h"
57
58 #undef mknod
59 #undef __xmknod
60
61 /*
62  * Internal routine to make a device node.
63  */
64 int
65 _sysio_mknod(struct pnode *pno, mode_t mode, dev_t dev)
66 {
67
68         if (pno->p_base->pb_ino)
69                 return -EEXIST;
70
71         /*
72          * Support only regular, character-special and fifos right now.
73          * (mode & S_IFMT) == 0 is the same as S_IFREG.
74          */
75         if (!(S_ISREG(mode) || S_ISCHR(mode) || S_ISFIFO(mode)))
76                 return -EINVAL;
77
78         if (IS_RDONLY(pno))
79                 return -EROFS;
80         return (*pno->p_parent->p_base->pb_ino->i_ops.inop_mknod)(pno,
81                                                                   mode,
82                                                                   dev);
83 }
84
85 int
86 PREPEND(__, SYSIO_INTERFACE_NAME(xmknod))(int __ver, 
87                                           const char *path, 
88                                           mode_t mode, 
89                                           dev_t *dev)
90 {
91         int     err;
92         struct intent intent;
93         struct pnode *pno;
94         SYSIO_INTERFACE_DISPLAY_BLOCK;
95
96         SYSIO_INTERFACE_ENTER;
97         if (__ver != _MKNOD_VER) {
98                 err = -ENOSYS;
99                 goto out;
100         }
101
102         mode &= ~(_sysio_umask & 0777);                 /* apply umask */
103
104         INTENT_INIT(&intent, INT_CREAT, &mode, NULL);
105         err = _sysio_namei(_sysio_cwd, path, ND_NEGOK, &intent, &pno);
106         if (err)
107                 goto out;
108
109         err = _sysio_permitted(pno->p_parent, W_OK);
110         if (err)
111                 goto error;
112         err = _sysio_mknod(pno, mode, *dev);
113 error:
114         P_RELE(pno);
115 out:
116         SYSIO_INTERFACE_RETURN(err ? -1 : 0, err);
117 }
118
119 #ifdef REDSTORM
120 #undef _xmknod
121 sysio_sym_weak_alias(PREPEND(__, SYSIO_INTERFACE_NAME(xmknod)), 
122                      PREPEND(_, SYSIO_INTERFACE_NAME(xmknod)))
123 #endif
124
125 static int
126 PREPEND(__, SYSIO_INTERFACE_NAME(mknod))(const char *path, 
127                                          mode_t mode, 
128                                          dev_t dev)
129 {
130
131         return PREPEND(__, SYSIO_INTERFACE_NAME(xmknod))(_MKNOD_VER, 
132                                                          path, 
133                                                          mode, 
134                                                          &dev);
135 }
136
137 sysio_sym_weak_alias(PREPEND(__, SYSIO_INTERFACE_NAME(mknod)),
138                      SYSIO_INTERFACE_NAME(mknod))