1 /*-*- Mode: C; c-basic-offset: 8; indent-tabs-mode: nil -*-*/
4 This file is part of systemd.
6 Copyright 2010-2012 Lennart Poettering
8 systemd is free software; you can redistribute it and/or modify it
9 under the terms of the GNU Lesser General Public License as published by
10 the Free Software Foundation; either version 2.1 of the License, or
11 (at your option) any later version.
13 systemd is distributed in the hope that it will be useful, but
14 WITHOUT ANY WARRANTY; without even the implied warranty of
15 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
16 Lesser General Public License for more details.
18 You should have received a copy of the GNU Lesser General Public License
19 along with systemd; If not, see <http://www.gnu.org/licenses/>.
23 #include <linux/magic.h>
24 #include <sys/statvfs.h>
27 #include "dirent-util.h"
31 #include "stat-util.h"
32 #include "string-util.h"
34 int is_symlink(const char *path
) {
39 if (lstat(path
, &info
) < 0)
42 return !!S_ISLNK(info
.st_mode
);
45 int is_dir(const char* path
, bool follow
) {
58 return !!S_ISDIR(st
.st_mode
);
61 int is_device_node(const char *path
) {
66 if (lstat(path
, &info
) < 0)
69 return !!(S_ISBLK(info
.st_mode
) || S_ISCHR(info
.st_mode
));
72 int dir_is_empty(const char *path
) {
73 _cleanup_closedir_
DIR *d
;
80 FOREACH_DIRENT(de
, d
, return -errno
)
86 bool null_or_empty(struct stat
*st
) {
89 if (S_ISREG(st
->st_mode
) && st
->st_size
<= 0)
92 /* We don't want to hardcode the major/minor of /dev/null,
93 * hence we do a simpler "is this a device node?" check. */
95 if (S_ISCHR(st
->st_mode
) || S_ISBLK(st
->st_mode
))
101 int null_or_empty_path(const char *fn
) {
106 if (stat(fn
, &st
) < 0)
109 return null_or_empty(&st
);
112 int null_or_empty_fd(int fd
) {
117 if (fstat(fd
, &st
) < 0)
120 return null_or_empty(&st
);
123 int path_is_read_only_fs(const char *path
) {
128 if (statvfs(path
, &st
) < 0)
131 if (st
.f_flag
& ST_RDONLY
)
134 /* On NFS, statvfs() might not reflect whether we can actually
135 * write to the remote share. Let's try again with
136 * access(W_OK) which is more reliable, at least sometimes. */
137 if (access(path
, W_OK
) < 0 && errno
== EROFS
)
143 int path_is_os_tree(const char *path
) {
149 /* We use /usr/lib/os-release as flag file if something is an OS */
150 p
= strjoina(path
, "/usr/lib/os-release");
155 /* Also check for the old location in /etc, just in case. */
156 p
= strjoina(path
, "/etc/os-release");
162 int files_same(const char *filea
, const char *fileb
) {
168 if (stat(filea
, &a
) < 0)
171 if (stat(fileb
, &b
) < 0)
174 return a
.st_dev
== b
.st_dev
&&
175 a
.st_ino
== b
.st_ino
;
178 bool is_fs_type(const struct statfs
*s
, statfs_f_type_t magic_value
) {
180 assert_cc(sizeof(statfs_f_type_t
) >= sizeof(s
->f_type
));
182 return F_TYPE_EQUAL(s
->f_type
, magic_value
);
185 int fd_check_fstype(int fd
, statfs_f_type_t magic_value
) {
188 if (fstatfs(fd
, &s
) < 0)
191 return is_fs_type(&s
, magic_value
);
194 int path_check_fstype(const char *path
, statfs_f_type_t magic_value
) {
195 _cleanup_close_
int fd
= -1;
197 fd
= open(path
, O_RDONLY
);
201 return fd_check_fstype(fd
, magic_value
);
204 bool is_temporary_fs(const struct statfs
*s
) {
205 return is_fs_type(s
, TMPFS_MAGIC
) ||
206 is_fs_type(s
, RAMFS_MAGIC
);
209 int fd_is_temporary_fs(int fd
) {
212 if (fstatfs(fd
, &s
) < 0)
215 return is_temporary_fs(&s
);