kvm tools: Fix dirent handling on dirent non-friendly filesystems
Some filesystems don't return valid d_type field in dirent
structure (e.g XFS). So we must handle this case properly.
Signed-off-by: Milan Kocian <milon@wq.cz>
Cc: Sasha Levin <levinsasha928@gmail.com>
[ penberg@kernel.org: cleanups ]
Signed-off-by: Pekka Enberg <penberg@kernel.org>
diff --git a/kvm-ipc.c b/kvm-ipc.c
index 7897519..b1c43dd 100644
--- a/kvm-ipc.c
+++ b/kvm-ipc.c
@@ -112,14 +112,38 @@
return s;
}
+static bool is_socket(const char *base_path, const struct dirent *dent)
+{
+ switch (dent->d_type) {
+ case DT_SOCK:
+ return true;
+
+ case DT_UNKNOWN: {
+ char path[PATH_MAX];
+ struct stat st;
+
+ sprintf(path, "%s/%s", base_path, dent->d_name);
+ if (stat(path, &st))
+ return false;
+
+ return S_ISSOCK(st.st_mode);
+ }
+ default:
+ return false;
+ }
+}
+
int kvm__enumerate_instances(int (*callback)(const char *name, int fd))
{
int sock;
DIR *dir;
struct dirent entry, *result;
int ret = 0;
+ const char *path;
- dir = opendir(kvm__get_dir());
+ path = kvm__get_dir();
+
+ dir = opendir(path);
if (!dir)
return -errno;
@@ -127,7 +151,7 @@
readdir_r(dir, &entry, &result);
if (result == NULL)
break;
- if (entry.d_type == DT_SOCK) {
+ if (is_socket(path, &entry)) {
ssize_t name_len = strlen(entry.d_name);
char *p;