2211 lines
65 KiB
Diff
2211 lines
65 KiB
Diff
From 1ba08909cdc395c14116d7cdf0f8f3442ba5e4c0 Mon Sep 17 00:00:00 2001
|
|
From: Bastien Nocera <hadess@hadess.net>
|
|
Date: Thu, 6 Aug 2009 22:55:47 +0100
|
|
Subject: [PATCH] Add AFC backend
|
|
|
|
Add a backend based on libiphone to access data on Apple's iPhone,
|
|
and iPod Touch.
|
|
|
|
Code by:
|
|
Patrick Walton <pcwalton@cs.ucla.edu>
|
|
Martin Szulecki <opensuse@sukimashita.com>
|
|
Nikias Bassen <nikias@gmx.li>
|
|
Bastien Nocera <hadess@hadess.net>
|
|
---
|
|
configure.ac | 26 +-
|
|
daemon/Makefile.am | 23 +
|
|
daemon/afc.mount.in | 7 +
|
|
daemon/gvfsbackendafc.c | 1226 ++++++++++++++++++++
|
|
daemon/gvfsbackendafc.h | 37 +
|
|
monitor/Makefile.am | 6 +-
|
|
monitor/afc/Makefile.am | 49 +
|
|
monitor/afc/afc.monitor | 5 +
|
|
monitor/afc/afcvolume.c | 336 ++++++
|
|
monitor/afc/afcvolume.h | 44 +
|
|
monitor/afc/afcvolumemonitor.c | 215 ++++
|
|
monitor/afc/afcvolumemonitor.h | 39 +
|
|
monitor/afc/afcvolumemonitordaemon.c | 31 +
|
|
.../org.gtk.Private.AfcVolumeMonitor.service.in | 4 +
|
|
14 files changed, 2046 insertions(+), 2 deletions(-)
|
|
create mode 100644 daemon/afc.mount.in
|
|
create mode 100644 daemon/gvfsbackendafc.c
|
|
create mode 100644 daemon/gvfsbackendafc.h
|
|
create mode 100644 monitor/afc/Makefile.am
|
|
create mode 100644 monitor/afc/afc.monitor
|
|
create mode 100644 monitor/afc/afcvolume.c
|
|
create mode 100644 monitor/afc/afcvolume.h
|
|
create mode 100644 monitor/afc/afcvolumemonitor.c
|
|
create mode 100644 monitor/afc/afcvolumemonitor.h
|
|
create mode 100644 monitor/afc/afcvolumemonitordaemon.c
|
|
create mode 100644 monitor/afc/org.gtk.Private.AfcVolumeMonitor.service.in
|
|
|
|
diff --git a/configure.ac b/configure.ac
|
|
index 7f1ec16..4675ae9 100644
|
|
--- a/configure.ac
|
|
+++ b/configure.ac
|
|
@@ -324,6 +324,28 @@ AC_SUBST(CDDA_CFLAGS)
|
|
|
|
AM_CONDITIONAL(USE_CDDA, [test "$msg_cdda" = "yes"])
|
|
|
|
+dnl *************************************************
|
|
+dnl *** Check if we should build with AFC backend ***
|
|
+dnl *************************************************
|
|
+AC_ARG_ENABLE(afc, [ --disable-afc build without AFC backend])
|
|
+msg_afc=no
|
|
+AFC_LIBS=
|
|
+AFC_CFLAGS=
|
|
+
|
|
+if test "x$enable_afc" != "xno" -a "x$msg_gudev" = "xyes" ; then
|
|
+ PKG_CHECK_EXISTS(libiphone-1.0 >= 0.9.2, msg_afc=yes)
|
|
+
|
|
+ if test "x$msg_afc" = "xyes"; then
|
|
+ PKG_CHECK_MODULES(AFC, libiphone-1.0 gudev-1.0)
|
|
+ AC_DEFINE(HAVE_AFC, 1, [Define to 1 if AFC is going to be built])
|
|
+ fi
|
|
+fi
|
|
+
|
|
+AC_SUBST(AFC_LIBS)
|
|
+AC_SUBST(AFC_CFLAGS)
|
|
+
|
|
+AM_CONDITIONAL(USE_AFC, [test "$msg_afc" = "yes"])
|
|
+
|
|
dnl *****************************************************
|
|
dnl *** Check if we should build with obexftp backend ***
|
|
dnl *****************************************************
|
|
@@ -695,6 +717,7 @@ monitor/proxy/Makefile
|
|
monitor/hal/Makefile
|
|
monitor/gdu/Makefile
|
|
monitor/gphoto2/Makefile
|
|
+monitor/afc/Makefile
|
|
gconf/Makefile
|
|
programs/Makefile
|
|
test/Makefile
|
|
@@ -712,7 +735,8 @@ echo "
|
|
FUSE support: $msg_fuse
|
|
CDDA support: $msg_cdda
|
|
Gphoto2 support: $msg_gphoto2
|
|
- archive support: $msg_archive
|
|
+ archive support: $msg_archive
|
|
+ AFC support: $msg_afc
|
|
GConf support: $msg_gconf
|
|
DNS-SD support: $msg_avahi
|
|
Build HAL volume monitor: $msg_hal (with fast init path: $have_hal_fast_init)
|
|
diff --git a/daemon/Makefile.am b/daemon/Makefile.am
|
|
index 733fa41..4c9e4af 100644
|
|
--- a/daemon/Makefile.am
|
|
+++ b/daemon/Makefile.am
|
|
@@ -101,6 +101,12 @@ mount_DATA += archive.mount
|
|
libexec_PROGRAMS += gvfsd-archive
|
|
endif
|
|
|
|
+mount_in_files += afc.mount.in
|
|
+if USE_AFC
|
|
+mount_DATA += afc.mount
|
|
+libexec_PROGRAMS += gvfsd-afc
|
|
+endif
|
|
+
|
|
EXTRA_DIST = gvfs-daemon.service.in $(mount_in_files) obexftp-marshal.list
|
|
|
|
DISTCLEANFILES = gvfs-daemon.service $(mount_DATA)
|
|
@@ -433,3 +439,20 @@ gvfsd_dav_LDADD = $(libraries) $(HTTP_LIBS)
|
|
if HAVE_AVAHI
|
|
gvfsd_dav_LDADD += $(top_builddir)/common/libgvfscommon-dnssd.la
|
|
endif
|
|
+
|
|
+gvfsd_afc_SOURCES = \
|
|
+ gvfsbackendafc.c gvfsbackendafc.h \
|
|
+ daemon-main.c daemon-main.h \
|
|
+ daemon-main-generic.c
|
|
+
|
|
+gvfsd_afc_CPPFLAGS = \
|
|
+ -DBACKEND_HEADER=gvfsbackendafc.h \
|
|
+ -DDEFAULT_BACKEND_TYPE=afc \
|
|
+ -DMAX_JOB_THREADS=1 \
|
|
+ $(AFC_CFLAGS) \
|
|
+ -DBACKEND_TYPES='"afc", G_VFS_TYPE_BACKEND_AFC,'
|
|
+
|
|
+gvfsd_afc_LDADD = \
|
|
+ $(libraries) \
|
|
+ $(AFC_LIBS)
|
|
+
|
|
diff --git a/daemon/afc.mount.in b/daemon/afc.mount.in
|
|
new file mode 100644
|
|
index 0000000..727d833
|
|
--- /dev/null
|
|
+++ b/daemon/afc.mount.in
|
|
@@ -0,0 +1,7 @@
|
|
+[Mount]
|
|
+Type=afc
|
|
+Exec=@libexecdir@/gvfsd-afc
|
|
+AutoMount=false
|
|
+Scheme=afc
|
|
+DefaultPort=1
|
|
+
|
|
diff --git a/daemon/gvfsbackendafc.c b/daemon/gvfsbackendafc.c
|
|
new file mode 100644
|
|
index 0000000..97d0a22
|
|
--- /dev/null
|
|
+++ b/daemon/gvfsbackendafc.c
|
|
@@ -0,0 +1,1226 @@
|
|
+/*
|
|
+ * gvfs/daemon/gvfsbackendafc.c
|
|
+ *
|
|
+ * Copyright (c) 2008 Patrick Walton <pcwalton@cs.ucla.edu>
|
|
+ */
|
|
+
|
|
+#include <config.h>
|
|
+
|
|
+#include <limits.h>
|
|
+#include <stdint.h>
|
|
+#include <stdio.h>
|
|
+#include <string.h>
|
|
+#include <stdlib.h>
|
|
+#include <unistd.h>
|
|
+#include <sys/types.h>
|
|
+#include <glib/gi18n.h>
|
|
+#include <errno.h>
|
|
+
|
|
+#define G_UDEV_API_IS_SUBJECT_TO_CHANGE
|
|
+#include <gudev/gudev.h>
|
|
+
|
|
+#include <libiphone/libiphone.h>
|
|
+#include <libiphone/lockdown.h>
|
|
+#include <libiphone/afc.h>
|
|
+
|
|
+#include "gvfsbackendafc.h"
|
|
+#include "gvfsjobopenforread.h"
|
|
+#include "gvfsjobread.h"
|
|
+#include "gvfsjobseekread.h"
|
|
+#include "gvfsjobopenforwrite.h"
|
|
+#include "gvfsjobwrite.h"
|
|
+#include "gvfsjobseekwrite.h"
|
|
+#include "gvfsjobsetdisplayname.h"
|
|
+#include "gvfsjobqueryinfo.h"
|
|
+#include "gvfsjobqueryfsinfo.h"
|
|
+#include "gvfsjobqueryattributes.h"
|
|
+#include "gvfsjobenumerate.h"
|
|
+#include "gvfsdaemonprotocol.h"
|
|
+#include "gvfsdaemonutils.h"
|
|
+
|
|
+#define G_VFS_BACKEND_AFC_MAX_FILE_SIZE G_MAXINT64
|
|
+int g_blocksize = 4096; /* assume this is the default block size */
|
|
+
|
|
+struct _GVfsBackendAfc {
|
|
+ GVfsBackend backend;
|
|
+
|
|
+ GUdevClient *client;
|
|
+
|
|
+ char uuid[41];
|
|
+ char *service;
|
|
+ char *model;
|
|
+ gboolean connected;
|
|
+
|
|
+ iphone_device_t dev;
|
|
+ afc_client_t afc_cli;
|
|
+};
|
|
+
|
|
+struct afc_error_mapping {
|
|
+ afc_error_t from;
|
|
+ GIOErrorEnum to;
|
|
+};
|
|
+
|
|
+static struct afc_error_mapping afc_error_to_g_io_error[] = {
|
|
+ { AFC_E_UNKNOWN_ERROR , G_IO_ERROR_FAILED },
|
|
+ { AFC_E_OP_HEADER_INVALID , G_IO_ERROR_FAILED },
|
|
+ { AFC_E_NO_RESOURCES , G_IO_ERROR_TOO_MANY_OPEN_FILES },
|
|
+ { AFC_E_READ_ERROR , G_IO_ERROR_NOT_DIRECTORY },
|
|
+ { AFC_E_WRITE_ERROR , G_IO_ERROR_FAILED },
|
|
+ { AFC_E_UNKNOWN_PACKET_TYPE , G_IO_ERROR_FAILED },
|
|
+ { AFC_E_INVALID_ARGUMENT , G_IO_ERROR_INVALID_ARGUMENT },
|
|
+ { AFC_E_OBJECT_NOT_FOUND , G_IO_ERROR_NOT_FOUND },
|
|
+ { AFC_E_OBJECT_IS_DIR , G_IO_ERROR_IS_DIRECTORY },
|
|
+ { AFC_E_DIR_NOT_EMPTY , G_IO_ERROR_NOT_EMPTY },
|
|
+ { AFC_E_PERM_DENIED , G_IO_ERROR_PERMISSION_DENIED },
|
|
+ { AFC_E_SERVICE_NOT_CONNECTED , G_IO_ERROR_HOST_NOT_FOUND },
|
|
+ { AFC_E_OP_TIMEOUT , G_IO_ERROR_TIMED_OUT },
|
|
+ { AFC_E_TOO_MUCH_DATA , G_IO_ERROR_FAILED },
|
|
+ { AFC_E_END_OF_DATA , G_IO_ERROR_FAILED },
|
|
+ { AFC_E_OP_NOT_SUPPORTED , G_IO_ERROR_NOT_SUPPORTED },
|
|
+ { AFC_E_OBJECT_EXISTS , G_IO_ERROR_EXISTS },
|
|
+ { AFC_E_OBJECT_BUSY , G_IO_ERROR_BUSY },
|
|
+ { AFC_E_NO_SPACE_LEFT , G_IO_ERROR_NO_SPACE },
|
|
+ { AFC_E_OP_WOULD_BLOCK , G_IO_ERROR_WOULD_BLOCK },
|
|
+ { AFC_E_IO_ERROR , G_IO_ERROR_FAILED },
|
|
+ { AFC_E_OP_INTERRUPTED , G_IO_ERROR_CANCELLED },
|
|
+ { AFC_E_OP_IN_PROGRESS , G_IO_ERROR_PENDING },
|
|
+ { AFC_E_INTERNAL_ERROR , G_IO_ERROR_FAILED },
|
|
+ { AFC_E_NOT_ENOUGH_DATA , G_IO_ERROR_CLOSED },
|
|
+ { AFC_E_MUX_ERROR , G_IO_ERROR_FAILED },
|
|
+ { -1 }
|
|
+};
|
|
+
|
|
+/**
|
|
+ * Tries to convert the AFC error value into a GIOError.
|
|
+ *
|
|
+ * @param client AFC client to retrieve status value from.
|
|
+ *
|
|
+ * @return errno value.
|
|
+ */
|
|
+static GIOErrorEnum
|
|
+g_io_error_from_afc_error (afc_error_t error)
|
|
+{
|
|
+ GIOErrorEnum res = G_IO_ERROR_FAILED;
|
|
+ int i = 0; gboolean found = FALSE;
|
|
+
|
|
+ while (afc_error_to_g_io_error[i++].from != -1)
|
|
+ {
|
|
+ if (afc_error_to_g_io_error[i].from == error)
|
|
+ {
|
|
+ res = afc_error_to_g_io_error[i++].to;
|
|
+ found = TRUE;
|
|
+ break;
|
|
+ }
|
|
+ }
|
|
+
|
|
+ if (!found)
|
|
+ g_warning ("Unknown AFC error (%d).\n", error);
|
|
+
|
|
+ return res;
|
|
+}
|
|
+
|
|
+G_DEFINE_TYPE(GVfsBackendAfc, g_vfs_backend_afc, G_VFS_TYPE_BACKEND)
|
|
+
|
|
+static void
|
|
+g_vfs_backend_afc_close_connection (GVfsBackendAfc *self)
|
|
+{
|
|
+ if (self->connected)
|
|
+ {
|
|
+ afc_client_free (self->afc_cli);
|
|
+ g_free (self->model);
|
|
+ self->model = NULL;
|
|
+ iphone_device_free (self->dev);
|
|
+ }
|
|
+ self->connected = FALSE;
|
|
+}
|
|
+
|
|
+static int
|
|
+g_vfs_backend_afc_check (afc_error_t cond, GVfsJob *job)
|
|
+{
|
|
+ GIOErrorEnum error;
|
|
+
|
|
+ if (G_LIKELY(cond == AFC_E_SUCCESS))
|
|
+ return 0;
|
|
+
|
|
+ error = g_io_error_from_afc_error (cond);
|
|
+ switch (cond)
|
|
+ {
|
|
+ case AFC_E_INTERNAL_ERROR:
|
|
+ g_vfs_job_failed (job, G_IO_ERROR, error,
|
|
+ _("Internal Apple File Control error"));
|
|
+ break;
|
|
+ case AFC_E_OBJECT_NOT_FOUND:
|
|
+ g_vfs_job_failed (job, G_IO_ERROR, error,
|
|
+ _("File does not exist"));
|
|
+ case AFC_E_DIR_NOT_EMPTY:
|
|
+ g_vfs_job_failed (job, G_IO_ERROR, error,
|
|
+ _("The directory is not empty"));
|
|
+ break;
|
|
+ case AFC_E_OP_TIMEOUT:
|
|
+ g_vfs_job_failed (job, G_IO_ERROR, error,
|
|
+ _("The device did not respond"));
|
|
+ break;
|
|
+ case AFC_E_NOT_ENOUGH_DATA:
|
|
+ g_vfs_job_failed (job, G_IO_ERROR, error,
|
|
+ _("The connection was interrupted"));
|
|
+ break;
|
|
+ case AFC_E_MUX_ERROR:
|
|
+ g_vfs_job_failed (job, G_IO_ERROR, error,
|
|
+ _("Invalid Apple File Control data received"));
|
|
+ break;
|
|
+ default:
|
|
+ g_vfs_job_failed (job, G_IO_ERROR, error,
|
|
+ _("Unhandled Apple File Control error (%d)"), cond);
|
|
+ break;
|
|
+ }
|
|
+
|
|
+ return 1;
|
|
+}
|
|
+
|
|
+static int
|
|
+g_vfs_backend_lockdownd_check (lockdownd_error_t cond, GVfsJob *job)
|
|
+{
|
|
+ if (G_LIKELY(cond == LOCKDOWN_E_SUCCESS))
|
|
+ return 0;
|
|
+
|
|
+ switch (cond)
|
|
+ {
|
|
+ case LOCKDOWN_E_INVALID_ARG:
|
|
+ g_vfs_job_failed (job, G_IO_ERROR, G_IO_ERROR_INVALID_ARGUMENT,
|
|
+ _("Lockdown Error: Invalid Argument"));
|
|
+ break;
|
|
+ default:
|
|
+ g_vfs_job_failed (job, G_IO_ERROR, G_IO_ERROR_FAILED,
|
|
+ _("Unhandled Lockdown error (%d)"), cond);
|
|
+ break;
|
|
+ }
|
|
+
|
|
+ return 1;
|
|
+}
|
|
+
|
|
+static int
|
|
+g_vfs_backend_iphone_check (iphone_error_t cond, GVfsJob *job)
|
|
+{
|
|
+ if (G_LIKELY(cond == IPHONE_E_SUCCESS))
|
|
+ return 0;
|
|
+
|
|
+ switch (cond)
|
|
+ {
|
|
+ case IPHONE_E_INVALID_ARG:
|
|
+ g_vfs_job_failed (job, G_IO_ERROR, G_IO_ERROR_INVALID_ARGUMENT,
|
|
+ _("iPhone Device Error: Invalid Argument"));
|
|
+ break;
|
|
+ case IPHONE_E_NO_DEVICE:
|
|
+ g_vfs_job_failed (job, G_IO_ERROR, G_IO_ERROR_FAILED,
|
|
+ _("iPhone Device Error: No device found. Make sure usbmuxd is set up correctly."));
|
|
+ default:
|
|
+ g_vfs_job_failed (job, G_IO_ERROR, G_IO_ERROR_FAILED,
|
|
+ _("Unhandled iPhone Device error (%d)"), cond);
|
|
+ break;
|
|
+ }
|
|
+
|
|
+ return 1;
|
|
+}
|
|
+
|
|
+static void
|
|
+_uevent_cb (GUdevClient *client,
|
|
+ const gchar *action,
|
|
+ GUdevDevice *device,
|
|
+ gpointer user_data)
|
|
+{
|
|
+ GVfsBackendAfc *afc_backend = G_VFS_BACKEND_AFC (user_data);
|
|
+ const char *uuid;
|
|
+
|
|
+ g_return_if_fail (afc_backend->uuid != NULL);
|
|
+
|
|
+ if (g_str_equal (action, "remove") == FALSE)
|
|
+ return;
|
|
+ uuid = g_udev_device_get_property (device, "ID_SERIAL_SHORT");
|
|
+ if (uuid == NULL ||
|
|
+ g_str_equal (uuid, afc_backend->uuid) == FALSE)
|
|
+ return;
|
|
+
|
|
+ g_print ("Shutting down AFC backend for device uuid %s\n", afc_backend->uuid);
|
|
+
|
|
+ g_vfs_backend_afc_close_connection (afc_backend);
|
|
+
|
|
+ /* TODO: need a cleaner way to force unmount ourselves */
|
|
+ exit (1);
|
|
+}
|
|
+
|
|
+/* Callback for mounting. */
|
|
+static void
|
|
+g_vfs_backend_afc_mount (GVfsBackend *backend,
|
|
+ GVfsJobMount *job,
|
|
+ GMountSpec *spec,
|
|
+ GMountSource *src,
|
|
+ gboolean automounting)
|
|
+{
|
|
+ const char *str;
|
|
+ char *tmp;
|
|
+ char *display_name;
|
|
+ int port, virtual_port;
|
|
+ GMountSpec *real_spec;
|
|
+ GVfsBackendAfc *self;
|
|
+ int retries;
|
|
+ iphone_error_t err;
|
|
+ lockdownd_client_t lockdown_cli = NULL;
|
|
+ const gchar * const subsystems[] = { "usb_endpoint", NULL };
|
|
+
|
|
+ self = G_VFS_BACKEND_AFC(backend);
|
|
+ self->connected = FALSE;
|
|
+ self->client = g_udev_client_new (subsystems);
|
|
+ g_signal_connect (G_OBJECT (self->client), "uevent",
|
|
+ G_CALLBACK (_uevent_cb), self);
|
|
+
|
|
+ /* setup afc */
|
|
+
|
|
+ str = g_mount_spec_get(spec, "host");
|
|
+ if (G_UNLIKELY(str == NULL))
|
|
+ {
|
|
+ g_vfs_job_failed (G_VFS_JOB (job),
|
|
+ G_IO_ERROR, G_IO_ERROR_INVALID_ARGUMENT,
|
|
+ _("Invalid mount spec"));
|
|
+ return;
|
|
+ }
|
|
+ if (G_UNLIKELY(sscanf(str, "%40s", &self->uuid) < 1))
|
|
+ {
|
|
+ g_vfs_job_failed (G_VFS_JOB(job), G_IO_ERROR, G_IO_ERROR_FAILED,
|
|
+ _("Invalid AFC location: must be in the form of "
|
|
+ "afc://uuid:port-number"));
|
|
+ return;
|
|
+ }
|
|
+
|
|
+ str = g_mount_spec_get (spec, "port");
|
|
+ if (str == NULL)
|
|
+ virtual_port = 1;
|
|
+ else
|
|
+ virtual_port = atoi (str);
|
|
+
|
|
+ /* set a generic display name */
|
|
+ if (virtual_port >= 2)
|
|
+ {
|
|
+ self->service = g_strdup_printf ("com.apple.afc%d", virtual_port);
|
|
+ display_name = g_strdup_printf (_("Service %d on Apple Mobile Device"),
|
|
+ virtual_port);
|
|
+ }
|
|
+ else
|
|
+ {
|
|
+ self->service = g_strdup ("com.apple.afc");
|
|
+ display_name = g_strdup_printf (_("Apple Mobile Device"));
|
|
+ }
|
|
+
|
|
+ g_vfs_backend_set_display_name (G_VFS_BACKEND(self), display_name);
|
|
+ g_free (display_name);
|
|
+
|
|
+ real_spec = g_mount_spec_new ("afc");
|
|
+ tmp = g_strdup_printf ("%40s", &self->uuid);
|
|
+ g_mount_spec_set (real_spec, "host", tmp);
|
|
+ g_free (tmp);
|
|
+
|
|
+ /* INFO: Don't ever set the DefaultPort again or everything goes crazy */
|
|
+ if (virtual_port != 1)
|
|
+ {
|
|
+ tmp = g_strdup_printf ("%d", virtual_port);
|
|
+ g_mount_spec_set (real_spec, "port", tmp);
|
|
+ g_free (tmp);
|
|
+ }
|
|
+
|
|
+ g_vfs_backend_set_mount_spec (G_VFS_BACKEND(self), real_spec);
|
|
+ g_mount_spec_unref (real_spec);
|
|
+
|
|
+ retries = 0;
|
|
+ do {
|
|
+ err = iphone_get_device_by_uuid(&self->dev, self->uuid);
|
|
+ if (err == IPHONE_E_SUCCESS)
|
|
+ break;
|
|
+ g_usleep (G_USEC_PER_SEC);
|
|
+ } while (retries++ < 10);
|
|
+
|
|
+ if (G_UNLIKELY(g_vfs_backend_iphone_check(err, G_VFS_JOB(job))))
|
|
+ goto out_destroy_service;
|
|
+ if (G_UNLIKELY(g_vfs_backend_lockdownd_check (lockdownd_client_new (self->dev,
|
|
+ &lockdown_cli),
|
|
+ G_VFS_JOB(job))))
|
|
+ {
|
|
+ goto out_destroy_dev;
|
|
+ }
|
|
+
|
|
+ /* try to use pretty device name */
|
|
+ if (LOCKDOWN_E_SUCCESS == lockdownd_get_device_name (lockdown_cli, &display_name))
|
|
+ {
|
|
+ if (display_name)
|
|
+ {
|
|
+ if (virtual_port >= 2)
|
|
+ {
|
|
+ /* translators:
|
|
+ * This is the device name, with the service being browsed in brackets, eg.:
|
|
+ * Alan Smithee's iPhone (Service 2 on Apple Mobile Device */
|
|
+ g_vfs_backend_set_display_name (G_VFS_BACKEND(self),
|
|
+ g_strdup_printf (_("%s (%s)"), display_name, self->service));
|
|
+ }
|
|
+ else
|
|
+ {
|
|
+ g_vfs_backend_set_display_name (G_VFS_BACKEND(self), display_name);
|
|
+ }
|
|
+ g_free (display_name);
|
|
+ }
|
|
+ }
|
|
+
|
|
+ if (G_UNLIKELY(g_vfs_backend_lockdownd_check (lockdownd_start_service (lockdown_cli,
|
|
+ self->service, &port),
|
|
+ G_VFS_JOB(job))))
|
|
+ {
|
|
+ goto out_destroy_lockdown;
|
|
+ }
|
|
+ if (G_UNLIKELY(g_vfs_backend_afc_check (afc_client_new (self->dev,
|
|
+ port, &self->afc_cli),
|
|
+ G_VFS_JOB(job))))
|
|
+ {
|
|
+ goto out_destroy_lockdown;
|
|
+ }
|
|
+
|
|
+ /* set correct fd icon spec name depending on device model */
|
|
+ self->model = afc_get_device_info_field (self->afc_cli, "Model");
|
|
+ if (G_UNLIKELY(self->model == NULL))
|
|
+ goto out_destroy_afc;
|
|
+
|
|
+ if (strstr(self->model, "iPod") != NULL)
|
|
+ {
|
|
+ g_vfs_backend_set_icon_name (G_VFS_BACKEND(self), "multimedia-player-apple-ipod-touch");
|
|
+ }
|
|
+ else
|
|
+ {
|
|
+ g_vfs_backend_set_icon_name (G_VFS_BACKEND(self), "phone-apple-iphone");
|
|
+ }
|
|
+
|
|
+ /* lockdown connection is not needed anymore */
|
|
+ lockdownd_client_free (lockdown_cli);
|
|
+
|
|
+ self->connected = TRUE;
|
|
+ g_vfs_job_succeeded (G_VFS_JOB(job));
|
|
+ return;
|
|
+
|
|
+out_destroy_afc:
|
|
+ afc_client_free (self->afc_cli);
|
|
+
|
|
+out_destroy_lockdown:
|
|
+ lockdownd_client_free (lockdown_cli);
|
|
+
|
|
+out_destroy_dev:
|
|
+ iphone_device_free (self->dev);
|
|
+
|
|
+out_destroy_service:
|
|
+ g_free (self->service);
|
|
+ g_free(self->model);
|
|
+}
|
|
+
|
|
+static void
|
|
+g_vfs_backend_afc_unmount (GVfsBackend *backend,
|
|
+ GVfsJobUnmount * job)
|
|
+{
|
|
+ GVfsBackendAfc *self;
|
|
+
|
|
+ self = G_VFS_BACKEND_AFC (backend);
|
|
+ g_vfs_backend_afc_close_connection (self);
|
|
+ g_vfs_job_succeeded (G_VFS_JOB(job));
|
|
+}
|
|
+
|
|
+/* Callback to open an existing file for reading. */
|
|
+static void
|
|
+g_vfs_backend_afc_open_for_read (GVfsBackend *backend,
|
|
+ GVfsJobOpenForRead *job,
|
|
+ const char *path)
|
|
+{
|
|
+ uint64_t fd = 0;
|
|
+ GVfsBackendAfc *self;
|
|
+
|
|
+ self = G_VFS_BACKEND_AFC(backend);
|
|
+ g_return_if_fail (self->connected);
|
|
+
|
|
+ if (G_UNLIKELY(g_vfs_backend_afc_check (afc_file_open (self->afc_cli,
|
|
+ path, AFC_FOPEN_RDONLY, &fd),
|
|
+ G_VFS_JOB(job))))
|
|
+ {
|
|
+ return;
|
|
+ }
|
|
+
|
|
+ g_vfs_job_open_for_read_set_handle (job, GUINT_TO_POINTER((gulong) fd));
|
|
+ g_vfs_job_open_for_read_set_can_seek (job, TRUE);
|
|
+ g_vfs_job_succeeded (G_VFS_JOB(job));
|
|
+
|
|
+ return;
|
|
+}
|
|
+
|
|
+/* Callback to open a nonexistent file for writing. */
|
|
+static void
|
|
+g_vfs_backend_afc_create (GVfsBackend *backend,
|
|
+ GVfsJobOpenForWrite *job,
|
|
+ const char *path,
|
|
+ GFileCreateFlags flags)
|
|
+{
|
|
+ uint64_t fd = 0;
|
|
+ GVfsBackendAfc *self;
|
|
+
|
|
+ self = G_VFS_BACKEND_AFC(backend);
|
|
+ g_return_if_fail (self->connected);
|
|
+
|
|
+ if (G_UNLIKELY(g_vfs_backend_afc_check (afc_file_open (self->afc_cli,
|
|
+ path, AFC_FOPEN_RW, &fd),
|
|
+ G_VFS_JOB(job))))
|
|
+ {
|
|
+ return;
|
|
+ }
|
|
+
|
|
+ g_vfs_job_open_for_write_set_handle (job, GUINT_TO_POINTER((gulong)fd));
|
|
+ g_vfs_job_open_for_write_set_can_seek (job, TRUE);
|
|
+ g_vfs_job_succeeded (G_VFS_JOB(job));
|
|
+
|
|
+ return;
|
|
+}
|
|
+
|
|
+/* Callback to open a possibly-existing file for writing. */
|
|
+static void
|
|
+g_vfs_backend_afc_append_to (GVfsBackend *backend,
|
|
+ GVfsJobOpenForWrite *job,
|
|
+ const char *path,
|
|
+ GFileCreateFlags flags)
|
|
+{
|
|
+ uint64_t fd = 0;
|
|
+ uint64_t off = 0;
|
|
+ GVfsBackendAfc *self;
|
|
+
|
|
+ self = G_VFS_BACKEND_AFC(backend);
|
|
+ g_return_if_fail (self->connected);
|
|
+
|
|
+ if (G_UNLIKELY(g_vfs_backend_afc_check (afc_file_open (self->afc_cli,
|
|
+ path, AFC_FOPEN_RW, &fd),
|
|
+ G_VFS_JOB(job))))
|
|
+ {
|
|
+ return;
|
|
+ }
|
|
+
|
|
+ if (G_UNLIKELY(g_vfs_backend_afc_check (afc_file_seek (self->afc_cli,
|
|
+ fd, 0, SEEK_END),
|
|
+ G_VFS_JOB(job))))
|
|
+ {
|
|
+ return;
|
|
+ }
|
|
+
|
|
+ if (G_UNLIKELY(g_vfs_backend_afc_check (afc_file_tell (self->afc_cli,
|
|
+ fd, &off),
|
|
+ G_VFS_JOB(job))))
|
|
+ {
|
|
+ return;
|
|
+ }
|
|
+
|
|
+ g_vfs_job_open_for_write_set_handle (job, GUINT_TO_POINTER((gulong)fd));
|
|
+ g_vfs_job_open_for_write_set_can_seek (job, TRUE);
|
|
+ g_vfs_job_open_for_write_set_initial_offset (job, off);
|
|
+ g_vfs_job_succeeded (G_VFS_JOB(job));
|
|
+
|
|
+ return;
|
|
+}
|
|
+
|
|
+static void
|
|
+g_vfs_backend_afc_replace (GVfsBackend *backend,
|
|
+ GVfsJobOpenForWrite *job,
|
|
+ const char *filename,
|
|
+ const char *etag,
|
|
+ gboolean make_backup,
|
|
+ GFileCreateFlags flags)
|
|
+{
|
|
+ uint64_t fd = 0;
|
|
+ GVfsBackendAfc *self;
|
|
+
|
|
+ self = G_VFS_BACKEND_AFC(backend);
|
|
+ g_return_if_fail(self->connected);
|
|
+
|
|
+ if (make_backup)
|
|
+ {
|
|
+ /* FIXME: implement! */
|
|
+ g_vfs_job_failed (G_VFS_JOB (job),
|
|
+ G_IO_ERROR,
|
|
+ G_IO_ERROR_CANT_CREATE_BACKUP,
|
|
+ _("Backups are not yet supported."));
|
|
+ return;
|
|
+ }
|
|
+
|
|
+ if (G_UNLIKELY(g_vfs_backend_afc_check (afc_file_open (self->afc_cli,
|
|
+ filename, AFC_FOPEN_WR, &fd),
|
|
+ G_VFS_JOB(job))))
|
|
+ {
|
|
+ return;
|
|
+ }
|
|
+
|
|
+ g_vfs_job_open_for_write_set_handle (job, GUINT_TO_POINTER((gulong)fd));
|
|
+ g_vfs_job_open_for_write_set_can_seek (job, TRUE);
|
|
+ g_vfs_job_succeeded (G_VFS_JOB(job));
|
|
+
|
|
+ return;
|
|
+}
|
|
+
|
|
+/* Callback to close a file that was previously opened for reading. */
|
|
+static void
|
|
+g_vfs_backend_afc_close_read (GVfsBackend *backend,
|
|
+ GVfsJobCloseRead *job,
|
|
+ GVfsBackendHandle handle)
|
|
+{
|
|
+ GVfsBackendAfc *self;
|
|
+ uint64_t fd = 0;
|
|
+
|
|
+ fd = GPOINTER_TO_UINT(handle);
|
|
+ g_return_if_fail (fd != 0);
|
|
+
|
|
+ self = G_VFS_BACKEND_AFC(backend);
|
|
+
|
|
+ if (self->connected)
|
|
+ afc_file_close (self->afc_cli, fd);
|
|
+
|
|
+ g_vfs_job_succeeded (G_VFS_JOB(job));
|
|
+}
|
|
+
|
|
+static void
|
|
+g_vfs_backend_afc_close_write (GVfsBackend *backend,
|
|
+ GVfsJobCloseWrite *job,
|
|
+ GVfsBackendHandle handle)
|
|
+{
|
|
+ GVfsBackendAfc *self;
|
|
+ uint64_t fd = 0;
|
|
+
|
|
+ fd = GPOINTER_TO_UINT(handle);
|
|
+ g_return_if_fail (fd != 0);
|
|
+
|
|
+ self = G_VFS_BACKEND_AFC(backend);
|
|
+
|
|
+ if (self->connected)
|
|
+ afc_file_close(self->afc_cli, fd);
|
|
+
|
|
+ g_vfs_job_succeeded (G_VFS_JOB(job));
|
|
+}
|
|
+
|
|
+static void
|
|
+g_vfs_backend_afc_read (GVfsBackend *backend,
|
|
+ GVfsJobRead *job,
|
|
+ GVfsBackendHandle handle,
|
|
+ char *buffer,
|
|
+ gsize req)
|
|
+{
|
|
+ guint32 nread = 0;
|
|
+ GVfsBackendAfc *self;
|
|
+ uint64_t fd = 0;
|
|
+
|
|
+ fd = GPOINTER_TO_UINT(handle);
|
|
+ g_return_if_fail (fd != 0);
|
|
+
|
|
+ self = G_VFS_BACKEND_AFC(backend);
|
|
+ g_return_if_fail (self->connected);
|
|
+
|
|
+ if (req > 0 &&
|
|
+ G_UNLIKELY(g_vfs_backend_afc_check (afc_file_read (self->afc_cli,
|
|
+ fd, buffer, req, &nread),
|
|
+ G_VFS_JOB(job))))
|
|
+ {
|
|
+ return;
|
|
+ }
|
|
+
|
|
+ g_vfs_job_read_set_size (job, nread);
|
|
+ g_vfs_job_succeeded (G_VFS_JOB(job));
|
|
+}
|
|
+
|
|
+static void
|
|
+g_vfs_backend_afc_write (GVfsBackend *backend,
|
|
+ GVfsJobWrite *job,
|
|
+ GVfsBackendHandle handle,
|
|
+ char *buffer,
|
|
+ gsize sz)
|
|
+{
|
|
+ guint32 nwritten = 0;
|
|
+ GVfsBackendAfc *self;
|
|
+ uint64_t fd = 0;
|
|
+
|
|
+ fd = GPOINTER_TO_UINT(handle);
|
|
+ g_return_if_fail (fd != 0);
|
|
+
|
|
+ self = G_VFS_BACKEND_AFC(backend);
|
|
+ g_return_if_fail (self->connected);
|
|
+
|
|
+ if (sz > 0 &&
|
|
+ G_UNLIKELY(g_vfs_backend_afc_check(afc_file_write (self->afc_cli,
|
|
+ fd, buffer, sz, &nwritten),
|
|
+ G_VFS_JOB(job))))
|
|
+ {
|
|
+ return;
|
|
+ }
|
|
+
|
|
+ g_vfs_job_write_set_written_size (job, nwritten);
|
|
+ g_vfs_job_succeeded (G_VFS_JOB(job));
|
|
+}
|
|
+
|
|
+static int
|
|
+g_vfs_backend_afc_seek (GVfsBackendAfc *self,
|
|
+ GVfsJob *job,
|
|
+ GVfsBackendHandle handle,
|
|
+ goffset offset,
|
|
+ GSeekType type)
|
|
+{
|
|
+ int afc_seek_type;
|
|
+ uint64_t fd = 0;
|
|
+
|
|
+ switch (type)
|
|
+ {
|
|
+ case G_SEEK_SET:
|
|
+ afc_seek_type = SEEK_SET;
|
|
+ break;
|
|
+ case G_SEEK_CUR:
|
|
+ afc_seek_type = SEEK_CUR;
|
|
+ break;
|
|
+ case G_SEEK_END:
|
|
+ afc_seek_type = SEEK_END;
|
|
+ break;
|
|
+ default:
|
|
+ g_vfs_job_failed(job, G_IO_ERROR, G_IO_ERROR_INVALID_ARGUMENT,
|
|
+ _("Invalid seek type"));
|
|
+ return 1;
|
|
+ }
|
|
+
|
|
+ fd = GPOINTER_TO_UINT(handle);
|
|
+
|
|
+ if (G_UNLIKELY(g_vfs_backend_afc_check (afc_file_seek (self->afc_cli,
|
|
+ fd, offset, afc_seek_type),
|
|
+ job)))
|
|
+ {
|
|
+ return 1;
|
|
+ }
|
|
+
|
|
+ return 0;
|
|
+}
|
|
+
|
|
+static void
|
|
+g_vfs_backend_afc_seek_on_read (GVfsBackend *backend,
|
|
+ GVfsJobSeekRead *job,
|
|
+ GVfsBackendHandle handle,
|
|
+ goffset offset,
|
|
+ GSeekType type)
|
|
+{
|
|
+ GVfsBackendAfc *self;
|
|
+
|
|
+ g_return_if_fail (handle != NULL);
|
|
+
|
|
+ self = G_VFS_BACKEND_AFC(backend);
|
|
+ g_return_if_fail (self->connected);
|
|
+
|
|
+ if (!g_vfs_backend_afc_seek (self, G_VFS_JOB(job), handle, offset, type))
|
|
+ {
|
|
+ g_vfs_job_seek_read_set_offset (job, offset);
|
|
+ g_vfs_job_succeeded (G_VFS_JOB(job));
|
|
+ }
|
|
+}
|
|
+
|
|
+static void
|
|
+g_vfs_backend_afc_seek_on_write (GVfsBackend *backend,
|
|
+ GVfsJobSeekWrite *job,
|
|
+ GVfsBackendHandle handle,
|
|
+ goffset offset,
|
|
+ GSeekType type)
|
|
+{
|
|
+ GVfsBackendAfc *self;
|
|
+
|
|
+ g_return_if_fail (handle != NULL);
|
|
+
|
|
+ self = G_VFS_BACKEND_AFC(backend);
|
|
+ g_return_if_fail (self->connected);
|
|
+
|
|
+ if (!g_vfs_backend_afc_seek (self, G_VFS_JOB(job), handle, offset, type))
|
|
+ {
|
|
+ g_vfs_job_seek_write_set_offset (job, offset);
|
|
+ g_vfs_job_succeeded (G_VFS_JOB(job));
|
|
+ }
|
|
+}
|
|
+
|
|
+static void
|
|
+g_vfs_backend_afc_set_info_from_afcinfo (GVfsBackendAfc *self,
|
|
+ GFileInfo *info,
|
|
+ char **afcinfo,
|
|
+ const char *basename,
|
|
+ GFileAttributeMatcher *matcher,
|
|
+ GFileQueryInfoFlags flags)
|
|
+{
|
|
+ GFileType type = G_FILE_TYPE_REGULAR;
|
|
+ GIcon *icon = NULL;
|
|
+ gchar *content_type = NULL;
|
|
+ char *display_name;
|
|
+ char *linktarget = NULL;
|
|
+ char **afctargetinfo = NULL;
|
|
+ int i;
|
|
+
|
|
+ /* get file attributes from info list */
|
|
+ for (i = 0; afcinfo[i]; i += 2)
|
|
+ {
|
|
+ if (afcinfo[i] == NULL)
|
|
+ continue;
|
|
+ if (g_str_equal (afcinfo[i], "st_size"))
|
|
+ {
|
|
+ g_file_info_set_size (info, atoll(afcinfo[i+1]));
|
|
+ } else if (g_str_equal (afcinfo[i], "st_blocks"))
|
|
+ {
|
|
+ g_file_info_set_attribute_uint64 (info, G_FILE_ATTRIBUTE_UNIX_BLOCKS, atoi(afcinfo[i+1]));
|
|
+ }
|
|
+ else if (g_str_equal (afcinfo[i], "st_ifmt"))
|
|
+ {
|
|
+ if (g_str_equal (afcinfo[i+1], "S_IFREG"))
|
|
+ {
|
|
+ type = G_FILE_TYPE_REGULAR;
|
|
+ }
|
|
+ else if (g_str_equal (afcinfo[i+1], "S_IFDIR"))
|
|
+ {
|
|
+ type = G_FILE_TYPE_DIRECTORY;
|
|
+ }
|
|
+ else if (g_str_equal (afcinfo[i+1], "S_IFLNK"))
|
|
+ {
|
|
+ type = G_FILE_TYPE_SYMBOLIC_LINK;
|
|
+ content_type = g_strdup ("inode/symlink");
|
|
+ }
|
|
+ else if (g_str_equal (afcinfo[i+1], "S_IFBLK"))
|
|
+ {
|
|
+ type = G_FILE_TYPE_SPECIAL;
|
|
+ content_type = g_strdup ("inode/blockdevice");
|
|
+ }
|
|
+ else if (g_str_equal (afcinfo[i+1], "S_IFCHR"))
|
|
+ {
|
|
+ type = G_FILE_TYPE_SPECIAL;
|
|
+ content_type = g_strdup ("inode/chardevice");
|
|
+ }
|
|
+ else if (g_str_equal (afcinfo[i+1], "S_IFIFO"))
|
|
+ {
|
|
+ type = G_FILE_TYPE_SPECIAL;
|
|
+ content_type = g_strdup ("inode/fifo");
|
|
+ }
|
|
+ else if (g_str_equal (afcinfo[i+1], "S_IFSOCK"))
|
|
+ {
|
|
+ type = G_FILE_TYPE_SPECIAL;
|
|
+ content_type = g_strdup ("inode/socket");
|
|
+ }
|
|
+ g_file_info_set_file_type (info, type);
|
|
+ }
|
|
+ else if (g_str_equal (afcinfo[i], "st_nlink"))
|
|
+ {
|
|
+ g_file_info_set_attribute_uint32 (info, G_FILE_ATTRIBUTE_UNIX_NLINK, atoi(afcinfo[i+1]));
|
|
+ }
|
|
+ else if (g_str_equal (afcinfo[i], "LinkTarget"))
|
|
+ {
|
|
+ linktarget = g_strdup (afcinfo[i+1]);
|
|
+ g_file_info_set_symlink_target (info, linktarget);
|
|
+ g_file_info_set_is_symlink (info, TRUE);
|
|
+ }
|
|
+ }
|
|
+
|
|
+ /* and set some additional info */
|
|
+ g_file_info_set_attribute_uint32 (info, G_FILE_ATTRIBUTE_UNIX_UID, getuid ());
|
|
+ g_file_info_set_attribute_uint32 (info, G_FILE_ATTRIBUTE_UNIX_GID, getgid ());
|
|
+
|
|
+ /*
|
|
+ * Maybe this icon stuff should be moved out into a generic function? It
|
|
+ * seems a little funny to put this in the backends.
|
|
+ */
|
|
+ if (g_file_attribute_matcher_matches (matcher, G_FILE_ATTRIBUTE_STANDARD_CONTENT_TYPE)
|
|
+ || g_file_attribute_matcher_matches (matcher, G_FILE_ATTRIBUTE_STANDARD_ICON))
|
|
+ {
|
|
+ if (type == G_FILE_TYPE_DIRECTORY)
|
|
+ {
|
|
+ content_type = g_strdup ("inode/directory");
|
|
+ icon = g_themed_icon_new ("folder");
|
|
+ }
|
|
+ else
|
|
+ {
|
|
+ if (content_type == NULL)
|
|
+ content_type = g_content_type_guess (basename, NULL, 0, NULL);
|
|
+ if (content_type)
|
|
+ {
|
|
+ icon = g_content_type_get_icon (content_type);
|
|
+ if (G_IS_THEMED_ICON(icon))
|
|
+ g_themed_icon_append_name (G_THEMED_ICON(icon), "text-x-generic");
|
|
+ }
|
|
+ }
|
|
+
|
|
+ if (content_type)
|
|
+ g_file_info_set_content_type (info, content_type);
|
|
+
|
|
+ if (icon == NULL)
|
|
+ icon = g_themed_icon_new ("text-x-generic");
|
|
+
|
|
+ g_file_info_set_icon (info, icon);
|
|
+ g_object_unref (icon);
|
|
+ }
|
|
+
|
|
+ g_free (content_type);
|
|
+
|
|
+ /* for symlinks to work we need to return GFileInfo for the linktarget */
|
|
+ if ((flags & G_FILE_QUERY_INFO_NOFOLLOW_SYMLINKS) == 0)
|
|
+ {
|
|
+ if (type == G_FILE_TYPE_SYMBOLIC_LINK)
|
|
+ {
|
|
+ /* query the linktarget instead and merge the file info of it */
|
|
+ if (AFC_E_SUCCESS == afc_get_file_info (self->afc_cli, linktarget, &afctargetinfo))
|
|
+ g_vfs_backend_afc_set_info_from_afcinfo (self, info, afctargetinfo, linktarget, matcher, flags);
|
|
+ if (afctargetinfo)
|
|
+ g_strfreev (afctargetinfo);
|
|
+ }
|
|
+ }
|
|
+
|
|
+ g_free (linktarget);
|
|
+
|
|
+ /* regardless of symlink recursion; still set the basename of the source */
|
|
+ g_file_info_set_name(info, basename);
|
|
+
|
|
+ /* handle root directory */
|
|
+ if (g_str_equal (basename, "/"))
|
|
+ display_name = g_strdup (g_vfs_backend_get_display_name (G_VFS_BACKEND(self)));
|
|
+ else
|
|
+ display_name = g_filename_display_name (basename);
|
|
+
|
|
+ if (g_file_attribute_matcher_matches (matcher, G_FILE_ATTRIBUTE_STANDARD_DISPLAY_NAME))
|
|
+ g_file_info_set_display_name (info, display_name);
|
|
+
|
|
+ if (g_file_attribute_matcher_matches (matcher, G_FILE_ATTRIBUTE_STANDARD_EDIT_NAME))
|
|
+ g_file_info_set_edit_name (info, display_name);
|
|
+
|
|
+ g_free (display_name);
|
|
+}
|
|
+
|
|
+/* Callback for iterating over a directory. */
|
|
+static void
|
|
+g_vfs_backend_afc_enumerate (GVfsBackend *backend,
|
|
+ GVfsJobEnumerate *job,
|
|
+ const char *path,
|
|
+ GFileAttributeMatcher *matcher,
|
|
+ GFileQueryInfoFlags flags)
|
|
+{
|
|
+ GFileInfo *info;
|
|
+ GVfsBackendAfc *self;
|
|
+ gboolean trailing_slash;
|
|
+ gchar *file_path;
|
|
+ char **ptr, **list = NULL;
|
|
+ char **afcinfo = NULL;
|
|
+
|
|
+ self = G_VFS_BACKEND_AFC(backend);
|
|
+ g_return_if_fail (self->connected);
|
|
+
|
|
+ if (G_UNLIKELY(g_vfs_backend_afc_check (afc_read_directory (self->afc_cli, path, &list),
|
|
+ G_VFS_JOB(job))))
|
|
+ {
|
|
+ return;
|
|
+ }
|
|
+
|
|
+ trailing_slash = g_str_has_suffix (path, "/");
|
|
+
|
|
+ for (ptr = list; *ptr; ptr++)
|
|
+ {
|
|
+ if (g_str_equal(*ptr, ".") || g_str_equal(*ptr, ".."))
|
|
+ continue;
|
|
+
|
|
+ if (!trailing_slash)
|
|
+ file_path = g_strdup_printf ("%s/%s", path, *ptr);
|
|
+ else
|
|
+ file_path = g_strdup_printf ("%s%s", path, *ptr);
|
|
+
|
|
+ /*
|
|
+ * This call might fail if the file in question is removed while we're
|
|
+ * iterating over the directory list. In that case, just don't include
|
|
+ * it in the list.
|
|
+ */
|
|
+ if (G_LIKELY(afc_get_file_info(self->afc_cli, file_path, &afcinfo) == AFC_E_SUCCESS))
|
|
+ {
|
|
+ info = g_file_info_new ();
|
|
+ g_vfs_backend_afc_set_info_from_afcinfo (self, info, afcinfo, *ptr, matcher, flags);
|
|
+ g_vfs_job_enumerate_add_info (job, info);
|
|
+ g_object_unref (G_OBJECT(info));
|
|
+ g_strfreev (afcinfo);
|
|
+ }
|
|
+
|
|
+ g_free (file_path);
|
|
+ }
|
|
+
|
|
+ g_strfreev (list);
|
|
+
|
|
+ g_vfs_job_enumerate_done (job);
|
|
+ g_vfs_job_succeeded (G_VFS_JOB(job));
|
|
+}
|
|
+
|
|
+static void
|
|
+g_vfs_backend_afc_query_info (GVfsBackend *backend,
|
|
+ GVfsJobQueryInfo *job,
|
|
+ const char *path,
|
|
+ GFileQueryInfoFlags flags,
|
|
+ GFileInfo *info,
|
|
+ GFileAttributeMatcher *matcher)
|
|
+{
|
|
+ GVfsBackendAfc *self;
|
|
+ const char *basename, *ptr;
|
|
+ char **afcinfo = NULL;
|
|
+
|
|
+ self = G_VFS_BACKEND_AFC(backend);
|
|
+ g_return_if_fail (self->connected);
|
|
+
|
|
+ if (G_UNLIKELY(g_vfs_backend_afc_check (afc_get_file_info (self->afc_cli, path, &afcinfo),
|
|
+ G_VFS_JOB(job))))
|
|
+ {
|
|
+ if (afcinfo)
|
|
+ g_strfreev(afcinfo);
|
|
+ return;
|
|
+ }
|
|
+
|
|
+ ptr = strrchr (path, '/');
|
|
+ if (ptr && ptr[1] != '\0')
|
|
+ basename = ptr + 1;
|
|
+ else
|
|
+ basename = path;
|
|
+
|
|
+ g_vfs_backend_afc_set_info_from_afcinfo (self, info, afcinfo, basename, matcher, flags);
|
|
+ if (afcinfo)
|
|
+ g_strfreev (afcinfo);
|
|
+
|
|
+ g_vfs_job_succeeded (G_VFS_JOB(job));
|
|
+}
|
|
+
|
|
+/*
|
|
+ * The following keys are currently known:
|
|
+ * Model: 'iPhone1,1'
|
|
+ * FSTotalBytes: storage capacity of drive
|
|
+ * FSFreeBytes: free space on drive
|
|
+ * FSBlockSize: block granularity
|
|
+ */
|
|
+static void
|
|
+g_vfs_backend_afc_query_fs_info (GVfsBackend *backend,
|
|
+ GVfsJobQueryFsInfo *job,
|
|
+ const char *path,
|
|
+ GFileInfo *info,
|
|
+ GFileAttributeMatcher *matcher)
|
|
+{
|
|
+ GVfsBackendAfc *self;
|
|
+ char **kvps, **ptr;
|
|
+ uint64_t totalspace = 0, freespace = 0;
|
|
+ int blocksize = 0;
|
|
+
|
|
+ self = G_VFS_BACKEND_AFC(backend);
|
|
+
|
|
+ g_file_info_set_attribute_string (info, G_FILE_ATTRIBUTE_FILESYSTEM_TYPE, "afc");
|
|
+
|
|
+ if (self->connected)
|
|
+ {
|
|
+ if (G_UNLIKELY(g_vfs_backend_afc_check (afc_get_device_info (self->afc_cli, &kvps), G_VFS_JOB(job))))
|
|
+ return;
|
|
+
|
|
+ for (ptr = kvps; *ptr; ptr++)
|
|
+ {
|
|
+ if (g_str_equal (*ptr, "FSTotalBytes"))
|
|
+ {
|
|
+ totalspace = g_ascii_strtoull (*(ptr+1), (char **) NULL, 10);
|
|
+ }
|
|
+ else if (g_str_equal (*ptr, "FSFreeBytes"))
|
|
+ {
|
|
+ freespace = g_ascii_strtoull (*(ptr+1), (char **) NULL, 10);
|
|
+ }
|
|
+ else if (g_str_equal (*ptr, "FSBlockSize"))
|
|
+ {
|
|
+ blocksize = atoi (*(ptr+1));
|
|
+ }
|
|
+ }
|
|
+
|
|
+ g_strfreev (kvps);
|
|
+
|
|
+ g_file_info_set_attribute_uint32 (info,
|
|
+ G_FILE_ATTRIBUTE_UNIX_BLOCK_SIZE,
|
|
+ (guint32) blocksize);
|
|
+ g_file_info_set_attribute_uint64 (info,
|
|
+ G_FILE_ATTRIBUTE_FILESYSTEM_SIZE,
|
|
+ (guint64) totalspace);
|
|
+ g_file_info_set_attribute_uint64 (info,
|
|
+ G_FILE_ATTRIBUTE_FILESYSTEM_FREE,
|
|
+ (guint64) freespace);
|
|
+ g_file_info_set_attribute_boolean (info,
|
|
+ G_FILE_ATTRIBUTE_FILESYSTEM_READONLY,
|
|
+ FALSE);
|
|
+ g_file_info_set_attribute_uint32 (info,
|
|
+ G_FILE_ATTRIBUTE_FILESYSTEM_USE_PREVIEW,
|
|
+ G_FILESYSTEM_PREVIEW_TYPE_IF_LOCAL);
|
|
+ }
|
|
+
|
|
+ g_vfs_job_succeeded (G_VFS_JOB(job));
|
|
+}
|
|
+
|
|
+static void
|
|
+g_vfs_backend_afc_set_display_name (GVfsBackend *backend,
|
|
+ GVfsJobSetDisplayName *job,
|
|
+ const char *filename,
|
|
+ const char *display_name)
|
|
+{
|
|
+ GVfsBackendAfc *self;
|
|
+
|
|
+ self = G_VFS_BACKEND_AFC(backend);
|
|
+ g_return_if_fail (self->connected);
|
|
+
|
|
+ if (G_UNLIKELY(g_vfs_backend_afc_check (afc_rename_path (self->afc_cli,
|
|
+ filename, display_name),
|
|
+ G_VFS_JOB(job))))
|
|
+ {
|
|
+ return;
|
|
+ }
|
|
+
|
|
+ g_vfs_job_set_display_name_set_new_path (job, display_name);
|
|
+
|
|
+ g_vfs_job_succeeded (G_VFS_JOB(job));
|
|
+}
|
|
+
|
|
+static void
|
|
+g_vfs_backend_afc_make_directory (GVfsBackend *backend,
|
|
+ GVfsJobMakeDirectory *job,
|
|
+ const char *path)
|
|
+{
|
|
+ GVfsBackendAfc *self;
|
|
+
|
|
+ self = G_VFS_BACKEND_AFC(backend);
|
|
+ g_return_if_fail(self->connected);
|
|
+
|
|
+ if (G_UNLIKELY(g_vfs_backend_afc_check (afc_make_directory (self->afc_cli,
|
|
+ path),
|
|
+ G_VFS_JOB(job))))
|
|
+ {
|
|
+ return;
|
|
+ }
|
|
+
|
|
+ g_vfs_job_succeeded (G_VFS_JOB(job));
|
|
+}
|
|
+
|
|
+static void
|
|
+g_vfs_backend_afc_make_symlink (GVfsBackend *backend,
|
|
+ GVfsJobMakeSymlink *job,
|
|
+ const char *filename,
|
|
+ const char *symlink_value)
|
|
+{
|
|
+ GVfsBackendAfc *self;
|
|
+
|
|
+ self = G_VFS_BACKEND_AFC(backend);
|
|
+ g_return_if_fail (self->connected);
|
|
+
|
|
+ if (G_UNLIKELY(g_vfs_backend_afc_check (afc_make_link (self->afc_cli,
|
|
+ AFC_SYMLINK, symlink_value, filename),
|
|
+ G_VFS_JOB(job))))
|
|
+ {
|
|
+ return;
|
|
+ }
|
|
+
|
|
+ g_vfs_job_succeeded (G_VFS_JOB(job));
|
|
+}
|
|
+
|
|
+static void
|
|
+g_vfs_backend_afc_move (GVfsBackend *backend,
|
|
+ GVfsJobMove *job,
|
|
+ const char *source,
|
|
+ const char *destination,
|
|
+ GFileCopyFlags flags,
|
|
+ GFileProgressCallback progress_callback,
|
|
+ gpointer progress_callback_data)
|
|
+{
|
|
+ GVfsBackendAfc *self;
|
|
+
|
|
+ self = G_VFS_BACKEND_AFC(backend);
|
|
+ g_return_if_fail(self->connected);
|
|
+
|
|
+ if (flags & G_FILE_COPY_BACKUP)
|
|
+ {
|
|
+ /* FIXME: implement! */
|
|
+ g_vfs_job_failed (G_VFS_JOB (job),
|
|
+ G_IO_ERROR,
|
|
+ G_IO_ERROR_CANT_CREATE_BACKUP,
|
|
+ _("Backups are not yet supported."));
|
|
+ return;
|
|
+ }
|
|
+
|
|
+ if (G_UNLIKELY(g_vfs_backend_afc_check (afc_rename_path (self->afc_cli,
|
|
+ source, destination),
|
|
+ G_VFS_JOB(job))))
|
|
+ {
|
|
+ return;
|
|
+ }
|
|
+
|
|
+ g_vfs_job_succeeded (G_VFS_JOB(job));
|
|
+}
|
|
+
|
|
+static void
|
|
+g_vfs_backend_afc_delete (GVfsBackend *backend,
|
|
+ GVfsJobDelete *job,
|
|
+ const char *filename)
|
|
+{
|
|
+ GVfsBackendAfc *self;
|
|
+
|
|
+ self = G_VFS_BACKEND_AFC(backend);
|
|
+ g_return_if_fail (self->connected);
|
|
+
|
|
+ if (G_UNLIKELY(g_vfs_backend_afc_check (afc_remove_path (self->afc_cli,
|
|
+ filename),
|
|
+ G_VFS_JOB(job))))
|
|
+ {
|
|
+ return;
|
|
+ }
|
|
+
|
|
+ g_vfs_job_succeeded (G_VFS_JOB(job));
|
|
+}
|
|
+
|
|
+
|
|
+static void
|
|
+g_vfs_backend_afc_finalize (GObject *obj)
|
|
+{
|
|
+ GVfsBackendAfc *self;
|
|
+
|
|
+ self = G_VFS_BACKEND_AFC(obj);
|
|
+ g_vfs_backend_afc_close_connection (self);
|
|
+
|
|
+ if (G_OBJECT_CLASS(g_vfs_backend_afc_parent_class)->finalize)
|
|
+ (*G_OBJECT_CLASS(g_vfs_backend_afc_parent_class)->finalize) (obj);
|
|
+}
|
|
+
|
|
+static void
|
|
+g_vfs_backend_afc_init (GVfsBackendAfc *self)
|
|
+{
|
|
+ if (g_getenv ("GVFS_DEBUG") != NULL)
|
|
+ {
|
|
+ /* enable full debugging */
|
|
+ iphone_set_debug_level (1);
|
|
+ iphone_set_debug_mask (DBGMASK_ALL);
|
|
+ }
|
|
+}
|
|
+
|
|
+static void
|
|
+g_vfs_backend_afc_class_init (GVfsBackendAfcClass *klass)
|
|
+{
|
|
+ GObjectClass *gobject_class = G_OBJECT_CLASS(klass);
|
|
+ GVfsBackendClass *backend_class = G_VFS_BACKEND_CLASS(klass);
|
|
+
|
|
+ gobject_class->finalize = g_vfs_backend_afc_finalize;
|
|
+
|
|
+ backend_class->mount = g_vfs_backend_afc_mount;
|
|
+ backend_class->unmount = g_vfs_backend_afc_unmount;
|
|
+ backend_class->open_for_read = g_vfs_backend_afc_open_for_read;
|
|
+ backend_class->close_read = g_vfs_backend_afc_close_read;
|
|
+ backend_class->read = g_vfs_backend_afc_read;
|
|
+ backend_class->seek_on_read = g_vfs_backend_afc_seek_on_read;
|
|
+ backend_class->create = g_vfs_backend_afc_create;
|
|
+ backend_class->append_to = g_vfs_backend_afc_append_to;
|
|
+ backend_class->replace = g_vfs_backend_afc_replace;
|
|
+ backend_class->close_write = g_vfs_backend_afc_close_write;
|
|
+ backend_class->write = g_vfs_backend_afc_write;
|
|
+ backend_class->seek_on_write = g_vfs_backend_afc_seek_on_write;
|
|
+ backend_class->enumerate = g_vfs_backend_afc_enumerate;
|
|
+ backend_class->query_info = g_vfs_backend_afc_query_info;
|
|
+ backend_class->query_fs_info = g_vfs_backend_afc_query_fs_info;
|
|
+ backend_class->make_directory = g_vfs_backend_afc_make_directory;
|
|
+ backend_class->delete = g_vfs_backend_afc_delete;
|
|
+ backend_class->make_symlink = g_vfs_backend_afc_make_symlink;
|
|
+ backend_class->move = g_vfs_backend_afc_move;
|
|
+ backend_class->set_display_name = g_vfs_backend_afc_set_display_name;
|
|
+}
|
|
+
|
|
+/*
|
|
+ * vim: sw=2 ts=8 cindent expandtab cinoptions=f0,>4,n2,{2,(0,^-2,t0 ai
|
|
+ */
|
|
diff --git a/daemon/gvfsbackendafc.h b/daemon/gvfsbackendafc.h
|
|
new file mode 100644
|
|
index 0000000..6291180
|
|
--- /dev/null
|
|
+++ b/daemon/gvfsbackendafc.h
|
|
@@ -0,0 +1,37 @@
|
|
+/*
|
|
+ * gvfs/daemon/gvfsbackendafc.h
|
|
+ *
|
|
+ * Copyright (c) 2008 Patrick Walton <pcwalton@ucla.edu>
|
|
+ */
|
|
+
|
|
+#ifndef GVFSBACKENDAFC_H
|
|
+#define GVFSBACKENDAFC_H
|
|
+
|
|
+#include <gvfsbackend.h>
|
|
+#include <gmountspec.h>
|
|
+
|
|
+G_BEGIN_DECLS
|
|
+
|
|
+#define G_VFS_TYPE_BACKEND_AFC (g_vfs_backend_afc_get_type())
|
|
+#define G_VFS_BACKEND_AFC(o) (G_TYPE_CHECK_INSTANCE_CAST((o), G_VFS_TYPE_BACKEND_AFC, GVfsBackendAfc))
|
|
+#define G_VFS_BACKEND_AFC_CLASS(k) (G_TYPE_CHECK_CLASS_CAST((k), G_VFS_TYPE_BACKEND_AFC, GVfsBackendAfcClass))
|
|
+#define G_VFS_IS_BACKEND_AFC(o) (G_TYPE_CHECK_INSTANCE_TYPE((o), G_VFS_TYPE_BACKEND_AFC))
|
|
+#define G_VFS_IS_BACKEND_AFC_CLASS(k) (G_TYPE_CHECK_CLASS_TYPE((k), G_VFS_TYPE_BACKEND_AFC))
|
|
+#define G_VFS_BACKEND_AFC_GET_CLASS(o) (G_TYPE_INSTANCE_GET_CLASS((o), G_VFS_TYPE_BACKEND_AFC, GVfsBackendAfcClass))
|
|
+
|
|
+typedef struct _GVfsBackendAfc GVfsBackendAfc;
|
|
+typedef struct _GVfsBackendAfcClass GVfsBackendAfcClass;
|
|
+
|
|
+struct _GVfsBackendAfcClass {
|
|
+ GVfsBackendClass parent_class;
|
|
+};
|
|
+
|
|
+GType g_vfs_backend_afc_get_type (void) G_GNUC_CONST;
|
|
+
|
|
+G_END_DECLS
|
|
+
|
|
+#endif /* GVFSBACKENDAFC_H */
|
|
+
|
|
+/*
|
|
+ * vim: sw=2 ts=8 cindent expandtab cinoptions=f0,>4,n2,{2,(0,^-2,t0 ai
|
|
+ */
|
|
diff --git a/monitor/Makefile.am b/monitor/Makefile.am
|
|
index f68423e..1dc984f 100644
|
|
--- a/monitor/Makefile.am
|
|
+++ b/monitor/Makefile.am
|
|
@@ -1,4 +1,4 @@
|
|
-
|
|
+DIST_SUBDIRS = proxy hal gdu gphoto2 afc
|
|
SUBDIRS = proxy
|
|
|
|
if USE_HAL
|
|
@@ -12,3 +12,7 @@ endif
|
|
if USE_GPHOTO2
|
|
SUBDIRS += gphoto2
|
|
endif
|
|
+
|
|
+if USE_AFC
|
|
+SUBDIRS += afc
|
|
+endif
|
|
diff --git a/monitor/afc/Makefile.am b/monitor/afc/Makefile.am
|
|
new file mode 100644
|
|
index 0000000..9b3b17c
|
|
--- /dev/null
|
|
+++ b/monitor/afc/Makefile.am
|
|
@@ -0,0 +1,49 @@
|
|
+NULL =
|
|
+
|
|
+gvfs_src_dir = $(top_srcdir)/@with_gvfs_source@
|
|
+
|
|
+libexec_PROGRAMS = gvfs-afc-volume-monitor
|
|
+
|
|
+gvfs_afc_volume_monitor_SOURCES = \
|
|
+ afcvolume.c afcvolume.h \
|
|
+ afcvolumemonitor.c afcvolumemonitor.h \
|
|
+ afcvolumemonitordaemon.c \
|
|
+ $(NULL)
|
|
+
|
|
+gvfs_afc_volume_monitor_CFLAGS = \
|
|
+ -DG_LOG_DOMAIN=\"GVFS-AFC\" \
|
|
+ -I$(top_srcdir)/common \
|
|
+ -I$(top_srcdir)/monitor/proxy \
|
|
+ $(GLIB_CFLAGS) \
|
|
+ $(AFC_CFLAGS) \
|
|
+ $(WARN_CFLAGS) \
|
|
+ -DGIO_MODULE_DIR=\"$(GIO_MODULE_DIR)\" \
|
|
+ -DGVFS_LOCALEDIR=\"$(localedir)\" \
|
|
+ -DG_DISABLE_DEPRECATED \
|
|
+ $(NULL)
|
|
+
|
|
+gvfs_afc_volume_monitor_LDADD = \
|
|
+ $(GLIB_LIBS) \
|
|
+ $(DBUS_LIBS) \
|
|
+ $(AFC_LIBS) \
|
|
+ $(top_srcdir)/common/libgvfscommon.la \
|
|
+ $(top_srcdir)/monitor/proxy/libgvfsproxyvolumemonitordaemon-noin.la \
|
|
+ $(NULL)
|
|
+
|
|
+remote_volume_monitorsdir = $(datadir)/gvfs/remote-volume-monitors
|
|
+remote_volume_monitors_DATA = afc.monitor
|
|
+
|
|
+servicedir = $(datadir)/dbus-1/services
|
|
+service_in_files = org.gtk.Private.AfcVolumeMonitor.service.in
|
|
+service_DATA = $(service_in_files:.service.in=.service)
|
|
+
|
|
+$(service_DATA): $(service_in_files) Makefile
|
|
+ @sed -e "s|\@libexecdir\@|$(libexecdir)|" $< > $@
|
|
+
|
|
+clean-local:
|
|
+ rm -f *~ *.loT $(BUILT_SOURCES) $(service_DATA)
|
|
+
|
|
+DISTCLEANFILES = $(service_DATA)
|
|
+
|
|
+EXTRA_DIST = $(service_in_files) afc.monitor
|
|
+
|
|
diff --git a/monitor/afc/afc.monitor b/monitor/afc/afc.monitor
|
|
new file mode 100644
|
|
index 0000000..1663573
|
|
--- /dev/null
|
|
+++ b/monitor/afc/afc.monitor
|
|
@@ -0,0 +1,5 @@
|
|
+[RemoteVolumeMonitor]
|
|
+Name=GProxyVolumeMonitorAfc
|
|
+DBusName=org.gtk.Private.AfcVolumeMonitor
|
|
+IsNative=false
|
|
+
|
|
diff --git a/monitor/afc/afcvolume.c b/monitor/afc/afcvolume.c
|
|
new file mode 100644
|
|
index 0000000..fe290a1
|
|
--- /dev/null
|
|
+++ b/monitor/afc/afcvolume.c
|
|
@@ -0,0 +1,336 @@
|
|
+/*
|
|
+ * gvfs/monitor/afc/afc-volume.c
|
|
+ *
|
|
+ * Copyright (c) 2008 Patrick Walton <pcwalton@cs.ucla.edu>
|
|
+ */
|
|
+
|
|
+#include <config.h>
|
|
+#include <string.h>
|
|
+#include <glib.h>
|
|
+#include <gio/gio.h>
|
|
+
|
|
+#include <libiphone/libiphone.h>
|
|
+#include <libiphone/lockdown.h>
|
|
+#include <libiphone/afc.h>
|
|
+
|
|
+#include "afcvolume.h"
|
|
+
|
|
+#define DEFAULT_SERVICE "com.apple.afc"
|
|
+
|
|
+struct _GVfsAfcVolume {
|
|
+ GObject parent;
|
|
+
|
|
+ GVolumeMonitor *monitor;
|
|
+
|
|
+ char *uuid;
|
|
+
|
|
+ char *name;
|
|
+ char *icon;
|
|
+ char *icon_fallback;
|
|
+};
|
|
+
|
|
+static void g_vfs_afc_volume_iface_init (GVolumeIface *iface);
|
|
+
|
|
+G_DEFINE_TYPE_EXTENDED(GVfsAfcVolume, g_vfs_afc_volume, G_TYPE_OBJECT, 0,
|
|
+ G_IMPLEMENT_INTERFACE(G_TYPE_VOLUME, g_vfs_afc_volume_iface_init))
|
|
+
|
|
+static void
|
|
+g_vfs_afc_volume_finalize (GObject *self_)
|
|
+{
|
|
+ GVfsAfcVolume *self;
|
|
+
|
|
+ self = G_VFS_AFC_VOLUME(self);
|
|
+
|
|
+ g_free (self->uuid);
|
|
+
|
|
+ g_free (self->name);
|
|
+ g_free (self->icon);
|
|
+ g_free (self->icon_fallback);
|
|
+
|
|
+ if (G_OBJECT_CLASS(g_vfs_afc_volume_parent_class)->finalize)
|
|
+ (*G_OBJECT_CLASS(g_vfs_afc_volume_parent_class)->finalize) (G_OBJECT(self));
|
|
+}
|
|
+
|
|
+static void
|
|
+g_vfs_afc_volume_init (GVfsAfcVolume *self)
|
|
+{
|
|
+ GVfsAfcVolume *afc_volume = G_VFS_AFC_VOLUME (self);
|
|
+
|
|
+ afc_volume->name = g_strdup ("iPhone");
|
|
+ afc_volume->icon = g_strdup ("phone-apple-iphone");
|
|
+}
|
|
+
|
|
+static void
|
|
+g_vfs_afc_volume_class_init (GVfsAfcVolumeClass *klass)
|
|
+{
|
|
+ GObjectClass *gobject_class;
|
|
+ gobject_class = G_OBJECT_CLASS(klass);
|
|
+ gobject_class->finalize = g_vfs_afc_volume_finalize;
|
|
+}
|
|
+
|
|
+static int
|
|
+_g_vfs_afc_volume_update_metadata (GVfsAfcVolume *self)
|
|
+{
|
|
+ iphone_device_t dev;
|
|
+ afc_client_t afc_cli;
|
|
+ lockdownd_client_t lockdown_cli = NULL;
|
|
+ iphone_error_t err;
|
|
+ guint retries;
|
|
+ char *model, *display_name;
|
|
+ int port;
|
|
+
|
|
+ retries = 0;
|
|
+ do {
|
|
+ err = iphone_get_device_by_uuid (&dev, self->uuid);
|
|
+ if (err == IPHONE_E_SUCCESS)
|
|
+ break;
|
|
+ g_usleep (G_USEC_PER_SEC);
|
|
+ } while (retries++ < 10);
|
|
+
|
|
+ if (err != IPHONE_E_SUCCESS)
|
|
+ return 0;
|
|
+
|
|
+ if (lockdownd_client_new (dev, &lockdown_cli) != LOCKDOWN_E_SUCCESS)
|
|
+ {
|
|
+ iphone_device_free (dev);
|
|
+ return 0;
|
|
+ }
|
|
+
|
|
+ /* try to use pretty device name */
|
|
+ if (lockdownd_get_device_name (lockdown_cli, &display_name) == LOCKDOWN_E_SUCCESS)
|
|
+ {
|
|
+ g_free (self->name);
|
|
+ self->name = display_name;
|
|
+ }
|
|
+
|
|
+ if (lockdownd_start_service (lockdown_cli, DEFAULT_SERVICE, &port) != LOCKDOWN_E_SUCCESS)
|
|
+ {
|
|
+ lockdownd_client_free (lockdown_cli);
|
|
+ iphone_device_free (dev);
|
|
+ return 0;
|
|
+ }
|
|
+
|
|
+ if (afc_client_new (dev, port, &afc_cli) == AFC_E_SUCCESS)
|
|
+ {
|
|
+ /* set correct fd icon spec name depending on device model */
|
|
+ model = afc_get_device_info_field (afc_cli, "Model");
|
|
+ if (model != NULL)
|
|
+ {
|
|
+ if(strstr(model, "iPod") != NULL)
|
|
+ {
|
|
+ g_free (self->icon);
|
|
+ self->icon = g_strdup ("multimedia-player-apple-ipod-touch");
|
|
+ }
|
|
+ g_free (model);
|
|
+ }
|
|
+ afc_client_free(afc_cli);
|
|
+ }
|
|
+
|
|
+ lockdownd_client_free (lockdown_cli);
|
|
+ iphone_device_free (dev);
|
|
+
|
|
+ return 1;
|
|
+}
|
|
+
|
|
+GVfsAfcVolume *
|
|
+g_vfs_afc_volume_new (GVolumeMonitor *monitor,
|
|
+ const char *uuid)
|
|
+{
|
|
+ GVfsAfcVolume *self;
|
|
+
|
|
+ self = G_VFS_AFC_VOLUME(g_object_new (G_VFS_TYPE_AFC_VOLUME, NULL));
|
|
+ self->monitor = monitor;
|
|
+ self->uuid = g_strdup (uuid);
|
|
+
|
|
+ /* Get mount information here */
|
|
+ if (!_g_vfs_afc_volume_update_metadata (self))
|
|
+ return NULL;
|
|
+
|
|
+ return self;
|
|
+}
|
|
+
|
|
+static char *
|
|
+g_vfs_afc_volume_get_name (GVolume *volume)
|
|
+{
|
|
+ GVfsAfcVolume *afc_volume = G_VFS_AFC_VOLUME (volume);
|
|
+ char *name;
|
|
+
|
|
+ name = g_strdup (afc_volume->name);
|
|
+
|
|
+ return name;
|
|
+}
|
|
+
|
|
+static GIcon *
|
|
+g_vfs_afc_volume_get_icon (GVolume *volume)
|
|
+{
|
|
+ GVfsAfcVolume *afc_volume = G_VFS_AFC_VOLUME (volume);
|
|
+ GIcon *icon;
|
|
+
|
|
+ icon = g_themed_icon_new_with_default_fallbacks (afc_volume->icon);
|
|
+
|
|
+ return icon;
|
|
+}
|
|
+
|
|
+static char *
|
|
+g_vfs_afc_volume_get_uuid (GVolume *volume)
|
|
+{
|
|
+ GVfsAfcVolume *afc_volume = G_VFS_AFC_VOLUME (volume);
|
|
+
|
|
+ return g_strdup (afc_volume->uuid);
|
|
+}
|
|
+
|
|
+static gboolean
|
|
+g_vfs_afc_volume_can_mount (GVolume *volume)
|
|
+{
|
|
+ return TRUE;
|
|
+}
|
|
+
|
|
+static gboolean
|
|
+g_vfs_afc_volume_should_automount (GVolume *volume)
|
|
+{
|
|
+ return TRUE;
|
|
+}
|
|
+
|
|
+
|
|
+typedef struct
|
|
+{
|
|
+ GVfsAfcVolume *enclosing_volume;
|
|
+ GAsyncReadyCallback callback;
|
|
+ GFile *root;
|
|
+ gpointer user_data;
|
|
+} ActivationMountOp;
|
|
+
|
|
+static void
|
|
+mount_callback (GObject *source_object,
|
|
+ GAsyncResult *res,
|
|
+ gpointer user_data)
|
|
+{
|
|
+ ActivationMountOp *data = user_data;
|
|
+ data->callback (G_OBJECT (data->enclosing_volume), res, data->user_data);
|
|
+ g_object_unref (data->root);
|
|
+ g_free (data);
|
|
+}
|
|
+
|
|
+static void
|
|
+g_vfs_afc_volume_mount (GVolume *volume,
|
|
+ GMountMountFlags flags,
|
|
+ GMountOperation *mount_operation,
|
|
+ GCancellable *cancellable,
|
|
+ GAsyncReadyCallback callback,
|
|
+ gpointer user_data)
|
|
+{
|
|
+ GVfsAfcVolume *afc_volume = G_VFS_AFC_VOLUME (volume);
|
|
+ ActivationMountOp *data;
|
|
+ GFile *root;
|
|
+ char *uri;
|
|
+
|
|
+ g_print ("g_vfs_afc_volume_mount (can_mount=%d uuid=%s)\n",
|
|
+ g_vfs_afc_volume_can_mount (volume),
|
|
+ afc_volume->uuid);
|
|
+
|
|
+ uri = g_strdup_printf ("afc://%s", afc_volume->uuid);
|
|
+ root = g_file_new_for_uri (uri);
|
|
+ g_free (uri);
|
|
+
|
|
+ data = g_new0 (ActivationMountOp, 1);
|
|
+ data->enclosing_volume = afc_volume;
|
|
+ data->callback = callback;
|
|
+ data->user_data = user_data;
|
|
+ data->root = root;
|
|
+
|
|
+ g_object_set_data_full (G_OBJECT(volume), "root", g_object_ref (root), g_object_unref);
|
|
+
|
|
+ g_file_mount_enclosing_volume (root,
|
|
+ 0,
|
|
+ mount_operation,
|
|
+ cancellable,
|
|
+ mount_callback,
|
|
+ data);
|
|
+}
|
|
+
|
|
+static gboolean
|
|
+g_vfs_afc_volume_mount_finish (GVolume *volume,
|
|
+ GAsyncResult *result,
|
|
+ GError **error)
|
|
+{
|
|
+ GFile *root;
|
|
+ gboolean res;
|
|
+
|
|
+ root = g_object_get_data (G_OBJECT (volume), "root");
|
|
+ res = g_file_mount_enclosing_volume_finish (root, result, error);
|
|
+
|
|
+ return res;
|
|
+}
|
|
+
|
|
+static char *
|
|
+g_vfs_afc_volume_get_identifier (GVolume *volume,
|
|
+ const char *kind)
|
|
+{
|
|
+ GVfsAfcVolume *afc_volume = G_VFS_AFC_VOLUME (volume);
|
|
+ char *id;
|
|
+
|
|
+ id = NULL;
|
|
+ if (strcmp (kind, G_VOLUME_IDENTIFIER_KIND_UUID) == 0)
|
|
+ id = g_strdup (afc_volume->uuid);
|
|
+
|
|
+ return id;
|
|
+}
|
|
+
|
|
+static char **
|
|
+g_vfs_afc_volume_enumerate_identifiers (GVolume *volume)
|
|
+{
|
|
+ GVfsAfcVolume *afc_volume = G_VFS_AFC_VOLUME (volume);
|
|
+ GPtrArray *res;
|
|
+
|
|
+ res = g_ptr_array_new ();
|
|
+
|
|
+ g_ptr_array_add (res,
|
|
+ g_strdup (G_VOLUME_IDENTIFIER_KIND_HAL_UDI));
|
|
+
|
|
+ if (afc_volume->uuid && *afc_volume->uuid != 0)
|
|
+ {
|
|
+ g_ptr_array_add (res,
|
|
+ g_strdup (G_VOLUME_IDENTIFIER_KIND_UUID));
|
|
+ }
|
|
+
|
|
+ /* Null-terminate */
|
|
+ g_ptr_array_add (res, NULL);
|
|
+
|
|
+ return (char **)g_ptr_array_free (res, FALSE);
|
|
+}
|
|
+
|
|
+static GFile *
|
|
+g_vfs_afc_volume_get_activation_root (GVolume *volume)
|
|
+{
|
|
+ GFile *root = g_object_get_data (G_OBJECT (volume), "root");
|
|
+
|
|
+ return g_object_ref (root);
|
|
+}
|
|
+
|
|
+static void
|
|
+g_vfs_afc_volume_iface_init (GVolumeIface *iface)
|
|
+{
|
|
+ iface->get_name = g_vfs_afc_volume_get_name;
|
|
+ iface->get_icon = g_vfs_afc_volume_get_icon;
|
|
+ iface->get_uuid = g_vfs_afc_volume_get_uuid;
|
|
+ iface->can_mount = g_vfs_afc_volume_can_mount;
|
|
+ iface->should_automount = g_vfs_afc_volume_should_automount;
|
|
+ iface->mount_fn = g_vfs_afc_volume_mount;
|
|
+ iface->mount_finish = g_vfs_afc_volume_mount_finish;
|
|
+ iface->eject = NULL;
|
|
+ iface->eject_finish = NULL;
|
|
+ iface->get_identifier = g_vfs_afc_volume_get_identifier;
|
|
+ iface->enumerate_identifiers = g_vfs_afc_volume_enumerate_identifiers;
|
|
+ iface->get_activation_root = g_vfs_afc_volume_get_activation_root;
|
|
+}
|
|
+
|
|
+gboolean g_vfs_afc_volume_has_uuid(GVfsAfcVolume *volume, const char *uuid)
|
|
+{
|
|
+ GVfsAfcVolume *afc_volume = G_VFS_AFC_VOLUME (volume);
|
|
+ g_return_val_if_fail (uuid != NULL, FALSE);
|
|
+ return (g_strcmp0 (afc_volume->uuid, uuid) == 0);
|
|
+}
|
|
+
|
|
+/*
|
|
+ * vim: sw=2 ts=8 cindent expandtab cinoptions=f0,>4,n2,{2,(0,^-2,t0 ai
|
|
+ */
|
|
diff --git a/monitor/afc/afcvolume.h b/monitor/afc/afcvolume.h
|
|
new file mode 100644
|
|
index 0000000..de24cd5
|
|
--- /dev/null
|
|
+++ b/monitor/afc/afcvolume.h
|
|
@@ -0,0 +1,44 @@
|
|
+/*
|
|
+ * gvfs/monitor/afc/afc-volume.h
|
|
+ *
|
|
+ * Copyright (c) 2008 Patrick Walton <pcwalton@cs.ucla.edu>
|
|
+ */
|
|
+
|
|
+#ifndef GVFS_MONITOR_AFC_AFC_VOLUME_H
|
|
+#define GVFS_MONITOR_AFC_AFC_VOLUME_H
|
|
+
|
|
+#include <glib-object.h>
|
|
+#include <gio/gio.h>
|
|
+
|
|
+#include "afcvolumemonitor.h"
|
|
+
|
|
+G_BEGIN_DECLS
|
|
+
|
|
+#define G_VFS_TYPE_AFC_VOLUME (g_vfs_afc_volume_get_type())
|
|
+#define G_VFS_AFC_VOLUME(o) (G_TYPE_CHECK_INSTANCE_CAST((o), G_VFS_TYPE_AFC_VOLUME, GVfsAfcVolume))
|
|
+#define G_VFS_AFC_VOLUME_CLASS(k) (G_TYPE_CHECK_CLASS_CAST((k), G_VFS_TYPE_AFC_VOLUME, GVfsAfcVolumeClass))
|
|
+#define G_VFS_IS_AFC_VOLUME(o) (G_TYPE_CHECK_INSTANCE_TYPE((o), G_VFS_TYPE_AFC_VOLUME))
|
|
+#define G_VFS_IS_AFC_VOLUME_CLASS(k) ((G_TYPE_CHECK_CLASS_TYPE((k), G_VFS_TYPE_AFC_VOLUME))
|
|
+#define G_VFS_AFC_VOLUME_GET_CLASS(o) (G_TYPE_INSTANCE_GET_CLASS((o), G_VFS_TYPE_AFC_VOLUME, GVfsAfcVolumeClass))
|
|
+
|
|
+typedef struct _GVfsAfcVolume GVfsAfcVolume;
|
|
+typedef struct _GVfsAfcVolumeClass GVfsAfcVolumeClass;
|
|
+
|
|
+struct _GVfsAfcVolumeClass {
|
|
+ GObjectClass parent_class;
|
|
+};
|
|
+
|
|
+GType g_vfs_afc_volume_get_type (void) G_GNUC_CONST;
|
|
+
|
|
+GVfsAfcVolume *g_vfs_afc_volume_new (GVolumeMonitor *monitor,
|
|
+ const char *uuid);
|
|
+
|
|
+gboolean g_vfs_afc_volume_has_uuid (GVfsAfcVolume *volume, const char *uuid);
|
|
+
|
|
+G_END_DECLS
|
|
+
|
|
+#endif /* GVFS_MONITOR_AFC_AFC_VOLUME_H */
|
|
+
|
|
+/*
|
|
+ * vim: sw=2 ts=8 cindent expandtab cinoptions=f0,>4,n2,{2,(0,^-2,t0 ai
|
|
+ */
|
|
diff --git a/monitor/afc/afcvolumemonitor.c b/monitor/afc/afcvolumemonitor.c
|
|
new file mode 100644
|
|
index 0000000..c98b603
|
|
--- /dev/null
|
|
+++ b/monitor/afc/afcvolumemonitor.c
|
|
@@ -0,0 +1,215 @@
|
|
+/*
|
|
+ * gvfs/monitor/afc/afc-volume-monitor.c
|
|
+ *
|
|
+ * Copyright (c) 2008 Patrick Walton <pcwalton@ucla.edu>
|
|
+ */
|
|
+
|
|
+#include <config.h>
|
|
+#include <glib.h>
|
|
+#include <gmodule.h>
|
|
+#include <gvfsproxyvolumemonitordaemon.h>
|
|
+#include <stdio.h>
|
|
+#include <gio/gio.h>
|
|
+#define G_UDEV_API_IS_SUBJECT_TO_CHANGE
|
|
+#include <gudev/gudev.h>
|
|
+#include "afcvolume.h"
|
|
+#include "afcvolumemonitor.h"
|
|
+
|
|
+struct _GVfsAfcVolumeMonitor {
|
|
+ GNativeVolumeMonitor parent;
|
|
+ GUdevClient *client;
|
|
+ GList *volumes;
|
|
+};
|
|
+
|
|
+G_DEFINE_TYPE(GVfsAfcVolumeMonitor, g_vfs_afc_volume_monitor, G_TYPE_VOLUME_MONITOR)
|
|
+
|
|
+static void
|
|
+g_vfs_afc_monitor_create_volume (GVfsAfcVolumeMonitor *self,
|
|
+ const char *uuid)
|
|
+{
|
|
+ GVfsAfcVolume *volume = NULL;
|
|
+
|
|
+ g_print ("creating volume for device uuid '%s'\n", uuid);
|
|
+
|
|
+ volume = g_vfs_afc_volume_new (G_VOLUME_MONITOR (self), uuid);
|
|
+ if (volume != NULL)
|
|
+ {
|
|
+ self->volumes = g_list_prepend (self->volumes, volume);
|
|
+ g_signal_emit_by_name (self, "volume-added", volume);
|
|
+ }
|
|
+}
|
|
+
|
|
+static GVfsAfcVolume *
|
|
+find_volume_by_uuid (GVfsAfcVolumeMonitor *self,
|
|
+ const char * uuid)
|
|
+{
|
|
+ GList *l;
|
|
+
|
|
+ for (l = self->volumes; l != NULL; l = l->next)
|
|
+ {
|
|
+ GVfsAfcVolume *volume = l->data;
|
|
+ if (volume && g_vfs_afc_volume_has_uuid (volume, uuid))
|
|
+ return volume;
|
|
+ }
|
|
+
|
|
+ return NULL;
|
|
+}
|
|
+
|
|
+static void
|
|
+g_vfs_afc_monitor_remove_volume (GVfsAfcVolumeMonitor *self,
|
|
+ const char *uuid)
|
|
+{
|
|
+ GVfsAfcVolume *volume = NULL;
|
|
+
|
|
+ volume = find_volume_by_uuid (self, uuid);
|
|
+ if (volume != NULL)
|
|
+ {
|
|
+ g_print ("removing volume for device uuid '%s'\n", uuid);
|
|
+ self->volumes = g_list_remove (self->volumes, volume);
|
|
+ g_signal_emit_by_name (self, "volume-removed", volume);
|
|
+ }
|
|
+}
|
|
+
|
|
+static void
|
|
+g_vfs_afc_monitor_uevent (GUdevClient *client,
|
|
+ const gchar *action,
|
|
+ GUdevDevice *device,
|
|
+ gpointer user_data)
|
|
+{
|
|
+ GVfsAfcVolumeMonitor *self;
|
|
+ const char *vendor, *devname, *uuid;
|
|
+
|
|
+ self = G_VFS_AFC_VOLUME_MONITOR(user_data);
|
|
+
|
|
+ /* Vendor is Apple? */
|
|
+ vendor = g_udev_device_get_property (device, "ID_VENDOR");
|
|
+ if (vendor == NULL ||
|
|
+ g_str_equal (vendor, "Apple_Inc.") == FALSE)
|
|
+ return;
|
|
+
|
|
+ /* Device is for end point 85? */
|
|
+ devname = g_udev_device_get_device_file (device);
|
|
+ if (devname == NULL ||
|
|
+ g_str_has_suffix (devname, "_ep85") == FALSE)
|
|
+ return;
|
|
+
|
|
+ /* Get us a UUID */
|
|
+ uuid = g_udev_device_get_property (device, "ID_SERIAL_SHORT");
|
|
+ if (uuid == NULL)
|
|
+ {
|
|
+ g_warning ("Could not get UUID for device '%s'", devname);
|
|
+ return;
|
|
+ }
|
|
+
|
|
+ if (g_str_equal (action, "add") != FALSE)
|
|
+ g_vfs_afc_monitor_create_volume (self, uuid);
|
|
+ else
|
|
+ g_vfs_afc_monitor_remove_volume (self, uuid);
|
|
+}
|
|
+
|
|
+static GObject *
|
|
+g_vfs_afc_volume_monitor_constructor (GType type, guint ncps,
|
|
+ GObjectConstructParam *cps)
|
|
+{
|
|
+ GVfsAfcVolumeMonitor *self;
|
|
+ GList *devices, *l;
|
|
+ const gchar * const subsystems[] = { "usb_endpoint", NULL };
|
|
+
|
|
+ /* Boilerplate code to chain from parent. */
|
|
+ self = G_VFS_AFC_VOLUME_MONITOR((*G_OBJECT_CLASS(g_vfs_afc_volume_monitor_parent_class)->constructor)(type, ncps, cps));
|
|
+
|
|
+ self->client = g_udev_client_new (subsystems);
|
|
+ g_signal_connect (G_OBJECT (self->client), "uevent",
|
|
+ G_CALLBACK (g_vfs_afc_monitor_uevent), self);
|
|
+
|
|
+ self->volumes = NULL;
|
|
+
|
|
+ devices = g_udev_client_query_by_subsystem (self->client, subsystems[0]);
|
|
+ for (l = devices; l != NULL; l = l->next)
|
|
+ {
|
|
+ GUdevDevice *device = l->data;
|
|
+ g_vfs_afc_monitor_uevent (self->client, "add", device, self);
|
|
+ g_object_unref (device);
|
|
+ }
|
|
+ g_list_free (devices);
|
|
+
|
|
+ g_print ("Volume monitor alive\n");
|
|
+
|
|
+ return G_OBJECT(self);
|
|
+}
|
|
+
|
|
+static void
|
|
+list_free (GList *objects)
|
|
+{
|
|
+ g_list_foreach (objects, (GFunc)g_object_unref, NULL);
|
|
+ g_list_free (objects);
|
|
+}
|
|
+
|
|
+static void
|
|
+g_vfs_afc_volume_monitor_finalize (GObject *_self)
|
|
+{
|
|
+ GVfsAfcVolumeMonitor *self;
|
|
+
|
|
+ self = G_VFS_AFC_VOLUME_MONITOR(_self);
|
|
+
|
|
+ if (self->volumes)
|
|
+ list_free (self->volumes);
|
|
+
|
|
+ if (self->client)
|
|
+ {
|
|
+ g_object_unref (self->client);
|
|
+ self->client = NULL;
|
|
+ }
|
|
+
|
|
+ if (G_OBJECT_CLASS(g_vfs_afc_volume_monitor_parent_class)->finalize)
|
|
+ (*G_OBJECT_CLASS(g_vfs_afc_volume_monitor_parent_class)->finalize)( G_OBJECT(self));
|
|
+}
|
|
+
|
|
+static GList *
|
|
+g_vfs_afc_volume_monitor_get_volumes (GVolumeMonitor *_self)
|
|
+{
|
|
+ GVfsAfcVolumeMonitor *self;
|
|
+ GList *l;
|
|
+
|
|
+ self = G_VFS_AFC_VOLUME_MONITOR (_self);
|
|
+
|
|
+ l = g_list_copy (self->volumes);
|
|
+ g_list_foreach (l, (GFunc)g_object_ref, NULL);
|
|
+
|
|
+ return l;
|
|
+}
|
|
+
|
|
+static gboolean
|
|
+g_vfs_afc_volume_monitor_is_supported (void)
|
|
+{
|
|
+ return TRUE;
|
|
+}
|
|
+
|
|
+static void
|
|
+g_vfs_afc_volume_monitor_class_init (GVfsAfcVolumeMonitorClass * klass)
|
|
+{
|
|
+ GObjectClass *gobject_class = G_OBJECT_CLASS(klass);
|
|
+ GVolumeMonitorClass *monitor_class = G_VOLUME_MONITOR_CLASS(klass);
|
|
+
|
|
+ gobject_class->constructor = g_vfs_afc_volume_monitor_constructor;
|
|
+ gobject_class->finalize = g_vfs_afc_volume_monitor_finalize;
|
|
+
|
|
+ monitor_class->get_volumes = g_vfs_afc_volume_monitor_get_volumes;
|
|
+ monitor_class->is_supported = g_vfs_afc_volume_monitor_is_supported;
|
|
+}
|
|
+
|
|
+static void
|
|
+g_vfs_afc_volume_monitor_init(GVfsAfcVolumeMonitor *self)
|
|
+{
|
|
+}
|
|
+
|
|
+GVolumeMonitor *
|
|
+g_vfs_afc_volume_monitor_new (void)
|
|
+{
|
|
+ return G_VOLUME_MONITOR(g_object_new (G_VFS_TYPE_AFC_VOLUME_MONITOR,
|
|
+ NULL));
|
|
+}
|
|
+
|
|
+/*
|
|
+ * vim: sw=2 ts=8 cindent expandtab cinoptions=f0,>4,n2,{2,(0,^-2,t0 ai
|
|
+ */
|
|
diff --git a/monitor/afc/afcvolumemonitor.h b/monitor/afc/afcvolumemonitor.h
|
|
new file mode 100644
|
|
index 0000000..0bd5f32
|
|
--- /dev/null
|
|
+++ b/monitor/afc/afcvolumemonitor.h
|
|
@@ -0,0 +1,39 @@
|
|
+/*
|
|
+ * gvfs/monitor/afc/afc-volume-monitor.h
|
|
+ *
|
|
+ * Copyright (c) 2008 Patrick Walton <pcwalton@ucla.edu>
|
|
+ */
|
|
+
|
|
+#ifndef AFC_VOLUME_MONITOR_H
|
|
+#define AFC_VOLUME_MONITOR_H
|
|
+
|
|
+#include <glib-object.h>
|
|
+#include <gio/gio.h>
|
|
+
|
|
+G_BEGIN_DECLS
|
|
+
|
|
+#define G_VFS_TYPE_AFC_VOLUME_MONITOR (g_vfs_afc_volume_monitor_get_type())
|
|
+#define G_VFS_AFC_VOLUME_MONITOR(o) (G_TYPE_CHECK_INSTANCE_CAST((o), G_VFS_TYPE_AFC_VOLUME_MONITOR, GVfsAfcVolumeMonitor))
|
|
+#define G_VFS_AFC_VOLUME_MONITOR_CLASS(k) (G_TYPE_CHECK_CLASS_CAST((k), G_VFS_TYPE_AFC_VOLUME_MONITOR, GVfsAfcVolumeMonitorClass))
|
|
+#define G_VFS_IS_AFC_VOLUME_MONITOR(o) (G_TYPE_CHECK_INSTANCE_TYPE((o), G_VFS_TYPE_AFC_VOLUME_MONITOR))
|
|
+#define G_VFS_IS_AFC_VOLUME_MONITOR_CLASS(k) (G_TYPE_CHECK_CLASS_TYPE((k), G_VFS_TYPE_AFC_VOLUME_MONITOR))
|
|
+#define G_VFS_AFC_VOLUME_MONITOR_GET_CLASS(o) (G_TYPE_INSTANCE_GET_CLASS((o), G_VFS_TYPE_AFC_VOLUME_MONITOR, GVfsAfcVolumeMonitorClass))
|
|
+
|
|
+typedef struct _GVfsAfcVolumeMonitor GVfsAfcVolumeMonitor;
|
|
+typedef struct _GVfsAfcVolumeMonitorClass GVfsAfcVolumeMonitorClass;
|
|
+
|
|
+struct _GVfsAfcVolumeMonitorClass {
|
|
+ GVolumeMonitorClass parent_class;
|
|
+};
|
|
+
|
|
+GType g_vfs_afc_volume_monitor_get_type (void) G_GNUC_CONST;
|
|
+
|
|
+GVolumeMonitor *g_vfs_afc_volume_monitor_new (void);
|
|
+
|
|
+G_END_DECLS
|
|
+
|
|
+#endif /* AFC_VOLUME_MONITOR_H */
|
|
+
|
|
+/*
|
|
+ * vim: sw=2 ts=8 cindent expandtab cinoptions=f0,>4,n2,{2,(0,^-2,t0 ai
|
|
+ */
|
|
diff --git a/monitor/afc/afcvolumemonitordaemon.c b/monitor/afc/afcvolumemonitordaemon.c
|
|
new file mode 100644
|
|
index 0000000..9c24a34
|
|
--- /dev/null
|
|
+++ b/monitor/afc/afcvolumemonitordaemon.c
|
|
@@ -0,0 +1,31 @@
|
|
+/*
|
|
+ * gvfs/monitor/afc/afc-volume-monitor-daemon.c
|
|
+ *
|
|
+ * Copyright (c) 2008-2009 Patrick Walton <pcwalton@ucla.edu>
|
|
+ * Copyright (c) 2009 Martin Szulecki <opensuse@sukimashita.com>
|
|
+ */
|
|
+
|
|
+#include <config.h>
|
|
+
|
|
+#include <glib.h>
|
|
+#include <glib/gi18n-lib.h>
|
|
+#include <gmodule.h>
|
|
+#include <gio/gio.h>
|
|
+
|
|
+#include <gvfsproxyvolumemonitordaemon.h>
|
|
+
|
|
+#include "afcvolumemonitor.h"
|
|
+
|
|
+int
|
|
+main (int argc, char *argv[])
|
|
+{
|
|
+ g_vfs_proxy_volume_monitor_daemon_init ();
|
|
+ return g_vfs_proxy_volume_monitor_daemon_main (argc,
|
|
+ argv,
|
|
+ "org.gtk.Private.AfcVolumeMonitor",
|
|
+ G_VFS_TYPE_AFC_VOLUME_MONITOR);
|
|
+}
|
|
+
|
|
+/*
|
|
+ * vim: sw=2 ts=8 cindent expandtab cinoptions=f0,>4,n2,{2,(0,^-2,t0 ai
|
|
+ */
|
|
diff --git a/monitor/afc/org.gtk.Private.AfcVolumeMonitor.service.in b/monitor/afc/org.gtk.Private.AfcVolumeMonitor.service.in
|
|
new file mode 100644
|
|
index 0000000..4e6bd33
|
|
--- /dev/null
|
|
+++ b/monitor/afc/org.gtk.Private.AfcVolumeMonitor.service.in
|
|
@@ -0,0 +1,4 @@
|
|
+[D-BUS Service]
|
|
+Name=org.gtk.Private.AfcVolumeMonitor
|
|
+Exec=@libexecdir@/gvfs-afc-volume-monitor
|
|
+
|
|
--
|
|
1.6.2.5
|
|
|