1 /* Copyright (C) 2000, 2001, 2002 Free Software Foundation, Inc.
2 This file is part of the GNU C Library.
4 The GNU C Library is free software; you can redistribute it and/or
5 modify it under the terms of the GNU Lesser General Public
6 License as published by the Free Software Foundation; either
7 version 2.1 of the License, or (at your option) any later version.
9 The GNU C Library is distributed in the hope that it will be useful,
10 but WITHOUT ANY WARRANTY; without even the implied warranty of
11 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
12 Lesser General Public License for more details.
14 You should have received a copy of the GNU Lesser General Public
15 License along with the GNU C Library; if not, write to the Free
16 Software Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA
28 #include <sys/statfs.h>
29 #include <bits/libc-lock.h>
30 #include "linux_fsinfo.h"
33 /* Mount point of the shared memory filesystem. */
40 /* This is the default directory. */
41 static const char defaultdir
[] = "/dev/shm/";
43 /* Protect the `mountpoint' variable above. */
44 __libc_once_define (static, once
);
47 /* Determine where the shmfs is mounted (if at all). */
57 /* The canonical place is /dev/shm. This is at least what the
58 documentation tells everybody to do. */
59 if (__statfs (defaultdir
, &f
) == 0 && f
.f_type
== SHMFS_SUPER_MAGIC
)
61 /* It is in the normal place. */
62 mountpoint
.dir
= (char *) defaultdir
;
63 mountpoint
.dirlen
= sizeof (defaultdir
) - 1;
68 /* OK, do it the hard way. Look through the /proc/mounts file and if
69 this does not exist through /etc/fstab to find the mount point. */
70 fp
= __setmntent ("/proc/mounts", "r");
71 if (__builtin_expect (fp
== NULL
, 0))
73 fp
= __setmntent (_PATH_MNTTAB
, "r");
74 if (__builtin_expect (fp
== NULL
, 0))
75 /* There is nothing we can do. Blind guesses are not helpful. */
79 /* Now read the entries. */
80 while ((mp
= __getmntent_r (fp
, &resmem
, buf
, sizeof buf
)) != NULL
)
81 /* The original name is "shm" but this got changed in early Linux
83 if (strcmp (mp
->mnt_type
, "tmpfs") == 0
84 || strcmp (mp
->mnt_type
, "shm") == 0)
86 /* Found it. There might be more than one place where the
87 filesystem is mounted but one is enough for us. */
90 /* First make sure this really is the correct entry. At least
91 some versions of the kernel give wrong information because
92 of the implicit mount of the shmfs for SysV IPC. */
93 if (__statfs (mp
->mnt_dir
, &f
) != 0 || f
.f_type
!= SHMFS_SUPER_MAGIC
)
96 namelen
= strlen (mp
->mnt_dir
);
99 /* Hum, maybe some crippled entry. Keep on searching. */
102 mountpoint
.dir
= (char *) malloc (namelen
+ 2);
103 if (mountpoint
.dir
!= NULL
)
105 char *cp
= __mempcpy (mountpoint
.dir
, mp
->mnt_dir
, namelen
);
109 mountpoint
.dirlen
= cp
- mountpoint
.dir
;
115 /* Close the stream. */
120 /* Open shared memory object. This implementation assumes the shmfs
121 implementation introduced in the late 2.3.x kernel series to be
122 available. Normally the filesystem will be mounted at /dev/shm but
123 we fall back on searching for the actual mount point should opening
126 shm_open (const char *name
, int oflag
, mode_t mode
)
132 /* Determine where the shmfs is mounted. */
133 __libc_once (once
, where_is_shmfs
);
135 /* If we don't know the mount points there is nothing we can do. Ever. */
136 if (mountpoint
.dir
== NULL
)
138 __set_errno (ENOSYS
);
142 /* Construct the filename. */
143 while (name
[0] == '/')
148 /* The name "/" is not supported. */
149 __set_errno (EINVAL
);
153 namelen
= strlen (name
);
154 fname
= (char *) alloca (mountpoint
.dirlen
+ namelen
+ 1);
155 __mempcpy (__mempcpy (fname
, mountpoint
.dir
, mountpoint
.dirlen
),
158 /* And get the file descriptor.
159 XXX Maybe we should test each descriptor whether it really is for a
160 file on the shmfs. If this is what should be done the whole function
161 should be revamped since we can determine whether shmfs is available
162 while trying to open the file, all in one turn. */
163 fd
= open (fname
, oflag
| O_NOFOLLOW
, mode
);
166 /* We got a descriptor. Now set the FD_CLOEXEC bit. */
167 int flags
= fcntl (fd
, F_GETFD
, 0);
169 if (__builtin_expect (flags
, 0) >= 0)
172 flags
= fcntl (fd
, F_SETFD
, flags
);
177 /* Something went wrong. We cannot return the descriptor. */
178 int save_errno
= errno
;
181 __set_errno (save_errno
);
189 /* Unlink a shared memory object. */
191 shm_unlink (const char *name
)
196 /* Determine where the shmfs is mounted. */
197 __libc_once (once
, where_is_shmfs
);
199 if (mountpoint
.dir
== NULL
)
201 /* We cannot find the shmfs. If `name' is really a shared
202 memory object it must have been created by another process
203 and we have no idea where that process found the mountpoint. */
204 __set_errno (ENOENT
);
208 /* Construct the filename. */
209 while (name
[0] == '/')
214 /* The name "/" is not supported. */
215 __set_errno (ENOENT
);
219 namelen
= strlen (name
);
220 fname
= (char *) alloca (mountpoint
.dirlen
+ namelen
+ 1);
221 __mempcpy (__mempcpy (fname
, mountpoint
.dir
, mountpoint
.dirlen
),
224 /* And remove the file. */
225 return unlink (fname
);
229 static void __attribute__ ((unused
))
232 if (mountpoint
.dir
!= defaultdir
)
233 free (mountpoint
.dir
);
237 /* Make sure the table is freed if we want to free everything before
239 text_set_element (__libc_subfreeres
, freeit
);