summaryrefslogtreecommitdiff
path: root/src
diff options
context:
space:
mode:
Diffstat (limited to 'src')
-rw-r--r--src/gsystem-file-tree-walk.c219
-rw-r--r--src/gsystem-file-tree-walk.h62
-rw-r--r--src/libgsystem.h1
3 files changed, 282 insertions, 0 deletions
diff --git a/src/gsystem-file-tree-walk.c b/src/gsystem-file-tree-walk.c
new file mode 100644
index 0000000..0d42479
--- /dev/null
+++ b/src/gsystem-file-tree-walk.c
@@ -0,0 +1,219 @@
+/* -*- mode: C; c-file-style: "gnu"; indent-tabs-mode: nil; -*-
+ *
+ * Copyright (C) 2014 Colin Walters <walters@verbum.org>
+ *
+ * This library is free software; you can redistribute it and/or
+ * modify it under the terms of the GNU Lesser General Public
+ * License as published by the Free Software Foundation; either
+ * version 2 of the License, or (at your option) any later version.
+ *
+ * This library is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
+ * Lesser General Public License for more details.
+ *
+ * You should have received a copy of the GNU Lesser General Public
+ * License along with this library; if not, write to the
+ * Free Software Foundation, Inc., 59 Temple Place - Suite 330,
+ * Boston, MA 02111-1307, USA.
+ */
+
+/**
+ * SECTION:gsfiletreewalk
+ * @title: GSFileTreeWalk
+ * @short_description: Recurse over a directory tree
+ *
+ * While #GFileEnumerator provides an API to iterate over one
+ * directory, in many cases one wants to operate recursively.
+ * This API is designed to do that.
+ *
+ * In addition, a #GSFileTreeWalk contains Unix-native API such as
+ * file descriptor based enumeration.
+ */
+
+#include "config.h"
+
+#define _GSYSTEM_NO_LOCAL_ALLOC
+#include "libgsystem.h"
+#include "gsystem-file-tree-walk.h"
+#include "gsystem-enum-types.h"
+#include <glib-unix.h>
+#include <string.h>
+#include <sys/stat.h>
+#include <dirent.h>
+#include <fcntl.h>
+
+/* Taken from systemd/src/shared/util.h */
+union dirent_storage {
+ struct dirent dent;
+ guint8 storage[offsetof(struct dirent, d_name) +
+ ((NAME_MAX + 1 + sizeof(long)) & ~(sizeof(long) - 1))];
+};
+
+typedef GObjectClass GSFileTreeWalkClass;
+
+struct _GSFileTreeWalk
+{
+ GObject parent;
+
+ GSFileTreeWalkFlags flags;
+ int origin_dfd;
+ struct stat origin_stbuf;
+
+ guint owns_dfd : 1;
+};
+
+G_DEFINE_TYPE (GSFileTreeWalk, gs_file_tree_walk, G_TYPE_OBJECT);
+
+enum
+{
+ PROP_0,
+ PROP_FLAGS,
+ N_PROPS
+};
+
+static GParamSpec *gs_file_tree_walk_pspecs[N_PROPS];
+
+static void
+gs_file_tree_walk_init (GSFileTreeWalk *self)
+{
+ self->origin_dfd = -1;
+}
+
+static void
+gs_file_tree_walk_finalize (GObject *object)
+{
+ GSFileTreeWalk *self = GS_FILE_TREE_WALK (object);
+
+ if (self->owns_dfd && self->origin_dfd != -1)
+ (void) close (self->origin_dfd);
+
+ if (G_OBJECT_CLASS (gs_file_tree_walk_class)->finalize != NULL)
+ G_OBJECT_CLASS (gs_file_tree_walk_class)->finalize (object);
+}
+
+static void
+gs_file_tree_walk_class_init (GSFileTreeWalkClass *class)
+{
+ /**
+ * GSFileTreeWalk:flags:
+ */
+ gs_file_tree_walk_pspecs[PROP_FLAGS] = g_param_spec_flags ("flags", "", "",
+ GS_TYPE_FILE_TREE_WALK_FLAGS,
+ G_PARAM_READWRITE | G_PARAM_CONSTRUCT_ONLY |
+ G_PARAM_STATIC_STRINGS);
+
+ g_object_class_install_properties (gobject_class, N_PROPS, gs_file_tree_walk_pspecs);
+}
+
+static void
+gs_file_tree_walk_set_property (GObject *object,
+ guint prop_id,
+ const GValue *value,
+ GParamSpec *pspec)
+{
+ GSFileTreeWalk *self = GS_FILE_TREE_WALK (object);
+
+ switch (prop_id)
+ {
+ case PROP_FLAGS:
+ self->flags = g_value_get_flags (value);
+ break;
+
+ default:
+ g_assert_not_reached ();
+ }
+}
+
+static void
+gs_file_tree_walk_get_property (GObject *object,
+ guint prop_id,
+ GValue *value,
+ GParamSpec *pspec)
+{
+ GSFileTreeWalk *self = GS_FILE_TREE_WALK (object);
+
+ switch (prop_id)
+ {
+ case PROP_FLAGS:
+ g_value_set_flags (value, self->flags);
+ break;
+
+ default:
+ g_assert_not_reached ();
+ }
+}
+
+/**
+ * gs_file_tree_walk_open:
+ * @path: Directory to enumerate
+ * @flags: Flags controlling enumeration
+ * @cancellable: Cancellable
+ * @error: Error
+ *
+ * Returns: (transfer full): A new directory enumerator
+ */
+GSFileTreeWalk *
+gs_file_tree_walk_open (GFile *path,
+ GSFileTreeWalkFlags flags,
+ GCancellable *cancellable,
+ GError **error)
+{
+ gs_unref_object GSFileTreeWalk *ftw = g_object_new (GS_TYPE_FILE_TREE_WALK, "flags", flags, NULL);
+
+ ftw->owns_dfd = TRUE;
+
+ if (!gs_file_open_dir_fd (path, &ftw->origin_dfd,
+ cancellable, error))
+ return NULL;
+
+ return g_object_ref (ftw);
+}
+
+/**
+ * gs_file_tree_walk_open_at:
+ * @dfd: Directory file descriptor
+ * @flags: Flags controlling enumeration
+ * @cancellable: Cancellable
+ * @error: Error
+ *
+ * Returns: (transfer full): A new directory enumerator
+ */
+GSFileTreeWalk *
+gs_file_tree_walk_open_at (int dfd,
+ GSFileTreeWalkFlags flags,
+ GCancellable *cancellable,
+ GError **error)
+{
+ gs_unref_object GSFileTreeWalk *ftw = g_object_new (GS_TYPE_FILE_TREE_WALK, "flags", flags, NULL);
+
+ ftw->origin_dfd = dfd;
+ if (fstat (ftw->origin_dfd, &ftw->origin_stbuf) != 0)
+ {
+ int errsv = errno;
+ g_set_error (error, G_IO_ERROR, G_IO_ERROR_FAILED,
+ "fstat: %s", g_strerror (errsv));
+ return NULL;
+ }
+
+ return g_object_ref (ftw);
+}
+
+/**
+ * gs_file_tree_walk_next:
+ * @self: Self
+ * @out_file_type: (out): File type, may be %G_FILE_TYPE_UNKNOWN
+ *
+ * Returns: %TRUE if there are more files to traverse
+ */
+gboolean
+gs_file_tree_walk_next (GSFileTreeWalk *self,
+ GFileType *out_file_type)
+{
+}
+
+int gs_file_tree_walk_get_dirfd (GSFileTreeWalk *self);
+
+const char *gs_file_tree_walk_get_name (GSFileTreeWalk *self);
+
+char *gs_file_tree_walk_get_relpath (GSFileTreeWalk *self);
diff --git a/src/gsystem-file-tree-walk.h b/src/gsystem-file-tree-walk.h
new file mode 100644
index 0000000..582ff7f
--- /dev/null
+++ b/src/gsystem-file-tree-walk.h
@@ -0,0 +1,62 @@
+/* -*- mode: C; c-file-style: "gnu"; indent-tabs-mode: nil; -*-
+ *
+ * Copyright (C) 2014 Colin Walters <walters@verbum.org>.
+ *
+ * This library is free software; you can redistribute it and/or
+ * modify it under the terms of the GNU Lesser General Public
+ * License as published by the Free Software Foundation; either
+ * version 2 of the License, or (at your option) any later version.
+ *
+ * This library is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
+ * Lesser General Public License for more details.
+ *
+ * You should have received a copy of the GNU Lesser General Public
+ * License along with this library; if not, write to the
+ * Free Software Foundation, Inc., 59 Temple Place - Suite 330,
+ * Boston, MA 02111-1307, USA.
+ */
+
+#ifndef __GSYSTEM_FILE_TREEWALK_H__
+#define __GSYSTEM_FILE_TREEWALK_H__
+
+#include <gio/gio.h>
+
+G_BEGIN_DECLS
+
+#define GS_TYPE_FILE_TREE_WALK (gs_file_tree_walk_get_type ())
+#define GS_FILE_TREE_WALK(o) (G_TYPE_CHECK_INSTANCE_CAST ((o), GS_TYPE_FILE_TREE_WALK, GSFileTreeWalk))
+#define GS_IS_FILE_TREE_WALK(o) (G_TYPE_CHECK_INSTANCE_TYPE ((o), GS_TYPE_FILE_TREE_WALK))
+
+typedef struct _GSFileTreeWalk GSFileTreeWalk;
+
+GType gs_file_tree_walk_get_type (void) G_GNUC_CONST;
+
+typedef enum {
+ GS_FILE_TREE_WALK_FLAG_DEPTH = 1 << 0,
+ GS_FILE_TREE_WALK_FLAG_NOXDEV = 1 << 1
+} GSFileTreeWalkFlags;
+
+GSFileTreeWalk *gs_file_tree_walk_open (GFile *path,
+ GSFileTreeWalkFlags flags,
+ GCancellable *cancellable,
+ GError **error);
+
+GSFileTreeWalk *gs_file_tree_walk_open_at (int dfd,
+ GSFileTreeWalkFlags flags,
+ GCancellable *cancellable,
+ GError **error);
+
+gboolean gs_file_tree_walk_next (GSFileTreeWalk *self,
+ GFileType *out_file_type);
+
+int gs_file_tree_walk_get_dirfd (GSFileTreeWalk *self);
+
+const char *gs_file_tree_walk_get_name (GSFileTreeWalk *self);
+
+char *gs_file_tree_walk_get_relpath (GSFileTreeWalk *self);
+
+G_END_DECLS
+
+#endif
diff --git a/src/libgsystem.h b/src/libgsystem.h
index 60884b6..59df669 100644
--- a/src/libgsystem.h
+++ b/src/libgsystem.h
@@ -34,6 +34,7 @@ G_BEGIN_DECLS
} G_STMT_END;
#include <gsystem-console.h>
+#include <gsystem-enum-types.h>
#include <gsystem-file-utils.h>
#include <gsystem-shutil.h>
#if GLIB_CHECK_VERSION(2,34,0)