make update-po
[findutils.git] / find / fstype.c
blobd2df661099612e6de5ff9ae2d3a67bd816a0de42
1 /* fstype.c -- determine type of filesystems that files are on
2 Copyright (C) 1990, 91, 92, 93, 94, 2000, 2004 Free Software Foundation, Inc.
4 This program is free software; you can redistribute it and/or modify
5 it under the terms of the GNU General Public License as published by
6 the Free Software Foundation; either version 2, or (at your option)
7 any later version.
9 This program 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
12 GNU General Public License for more details.
14 You should have received a copy of the GNU General Public License
15 along with this program; if not, write to the Free Software
16 Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301,
17 USA.
20 /* Written by David MacKenzie <djm@gnu.org>.
22 * Converted to use gnulib's read_file_system_list()
23 * by James Youngman <jay@gnu.org> (which saves a lot
24 * of manual hacking of configure.in).
28 #include <config.h>
29 #include <errno.h>
30 #include <assert.h>
31 #include <stdbool.h>
33 #ifdef HAVE_SYS_TYPES_H
34 #include <sys/types.h>
35 #endif
37 /* The presence of unistd.h is assumed by gnulib these days, so we
38 * might as well assume it too.
40 #include <unistd.h>
42 #ifdef HAVE_SYS_MNTIO_H
43 #ifdef HAVE_FCNTL_H
44 #include <fcntl.h>
45 #endif
46 #include <sys/mntio.h>
47 #endif
48 #ifdef HAVE_SYS_MKDEV_H
49 #include <sys/mkdev.h>
50 #endif
52 #ifdef STDC_HEADERS
53 #include <stdlib.h>
54 #else
55 extern int errno;
56 #endif
58 #include "defs.h"
59 #include "../gnulib/lib/dirname.h"
60 #include "xalloc.h"
61 #include "modetype.h"
63 /* Need declaration of function `xstrtoumax' */
64 #include "../gnulib/lib/xstrtol.h"
66 #include "extendbuf.h"
67 #include "mountlist.h"
71 #if ENABLE_NLS
72 # include <libintl.h>
73 # define _(Text) gettext (Text)
74 #else
75 # define _(Text) Text
76 #endif
77 #ifdef gettext_noop
78 # define N_(String) gettext_noop (String)
79 #else
80 /* See locate.c for explanation as to why not use (String) */
81 # define N_(String) String
82 #endif
84 static char *filesystem_type_uncached PARAMS((const struct stat *statp));
87 /* Get MNTTYPE_IGNORE if it is available. */
88 #if HAVE_MNTENT_H
89 # include <mntent.h>
90 #endif
91 #if HAVE_SYS_MNTTAB_H
92 # include <stdio.h>
93 # include <sys/mnttab.h>
94 #endif
100 static void
101 free_file_system_list(struct mount_entry *p)
103 while (p)
105 struct mount_entry *pnext = p->me_next;
107 free(p->me_devname);
108 free(p->me_mountdir);
110 if(p->me_type_malloced)
111 free(p->me_type);
112 p->me_next = NULL;
113 free(p);
114 p = pnext;
121 #ifdef AFS
122 #include <netinet/in.h>
123 #include <afs/venus.h>
124 #if __STDC__
125 /* On SunOS 4, afs/vice.h defines this to rely on a pre-ANSI cpp. */
126 #undef _VICEIOCTL
127 #define _VICEIOCTL(id) ((unsigned int ) _IOW('V', id, struct ViceIoctl))
128 #endif
129 #ifndef _IOW
130 /* AFS on Solaris 2.3 doesn't get this definition. */
131 #include <sys/ioccom.h>
132 #endif
134 static int
135 in_afs (char *path)
137 static char space[2048];
138 struct ViceIoctl vi;
140 vi.in_size = 0;
141 vi.out_size = sizeof (space);
142 vi.out = space;
144 if (pioctl (path, VIOC_FILE_CELL_NAME, &vi, 1)
145 && (errno == EINVAL || errno == ENOENT))
146 return 0;
147 return 1;
149 #endif /* AFS */
151 /* Nonzero if the current filesystem's type is known. */
152 static int fstype_known = 0;
154 /* Return a static string naming the type of filesystem that the file PATH,
155 described by STATP, is on.
156 RELPATH is the file name relative to the current directory.
157 Return "unknown" if its filesystem type is unknown. */
159 char *
160 filesystem_type (const struct stat *statp)
162 static char *current_fstype = NULL;
163 static dev_t current_dev;
165 if (current_fstype != NULL)
167 if (fstype_known && statp->st_dev == current_dev)
168 return current_fstype; /* Cached value. */
169 free (current_fstype);
171 current_dev = statp->st_dev;
172 current_fstype = filesystem_type_uncached (statp);
173 return current_fstype;
176 static int
177 set_fstype_devno(struct mount_entry *p)
179 struct stat stbuf;
181 if (p->me_dev == (dev_t)-1)
183 if (0 == (options.xstat)(p->me_mountdir, &stbuf))
185 p->me_dev = stbuf.st_dev;
186 return 0;
188 else
190 return -1;
193 return 0; /* not needed */
197 /* Return a newly allocated string naming the type of filesystem that the
198 file PATH, described by STATP, is on.
199 RELPATH is the file name relative to the current directory.
200 Return "unknown" if its filesystem type is unknown. */
202 static char *
203 filesystem_type_uncached (const struct stat *statp)
205 struct mount_entry *entries, *entry;
206 char *type;
208 #ifdef AFS
209 if (in_afs(path))
211 fstype_known = 1;
212 return xstrdup("afs");
214 #endif
216 entries = read_file_system_list(true);
217 for (type=NULL, entry=entries; entry; entry=entry->me_next)
219 #ifdef MNTTYPE_IGNORE
220 if (!strcmp (entry->me_type, MNTTYPE_IGNORE))
221 continue;
222 #endif
223 set_fstype_devno(entry);
224 if (entry->me_dev == statp->st_dev)
225 type = xstrdup(entry->me_type);
227 free_file_system_list(entries);
229 /* Don't cache unknown values. */
230 fstype_known = (type != NULL);
232 return type ? type : xstrdup(_("unknown"));
236 char *
237 get_mounted_filesystems (void)
239 char *result = NULL;
240 size_t alloc_size = 0u;
241 size_t used = 0u;
242 struct mount_entry *entries, *entry;
244 entries = read_file_system_list(false);
245 for (entry=entries; entry; entry=entry->me_next)
247 size_t len;
249 #ifdef MNTTYPE_IGNORE
250 if (!strcmp (entry->me_type, MNTTYPE_IGNORE))
251 continue;
252 #endif
253 set_fstype_devno(entry);
255 len = strlen(entry->me_mountdir) + 1;
256 result = extendbuf(result, used+len, &alloc_size);
257 strcpy(&result[used], entry->me_mountdir);
258 used += len; /* len already includes one for the \0 */
261 free_file_system_list(entries);
262 return result;
266 dev_t *
267 get_mounted_devices (size_t *n)
269 size_t alloc_size = 0u;
270 size_t used = 0u;
271 struct mount_entry *entries, *entry;
272 dev_t *result = NULL;
274 for (entry = entries = read_file_system_list(false);
275 entry;
276 entry = entry->me_next)
278 result = extendbuf(result, sizeof(dev_t)*(used+1), &alloc_size);
279 set_fstype_devno(entry);
280 result[used] = entry->me_dev;
281 ++used;
283 free_file_system_list(entries);
284 *n = used;
285 return result;