This library is not a complete NBD client/server, although it has been
designed with this goal in mind. It currently only supports the
read-only operations (adding write operations shouldn't be difficult),
which is enough to redirect a block device for a cdrom drive or a
read-only device.
It uses GIO as its core, to provide a consistant and modern API, that
could be easiliy binded. However, GIO currently doesn't offer concurrent
IO operations on stream. If this library becomes a seperate project, it
might be worthwile to define only a simple interface for the NbdExport
object to let various backend handled concurrent operations.
---
configure.ac | 1 +
gtk/Makefile.am | 2 +-
gtk/nbd/Makefile.am | 46 +++
gtk/nbd/nbd-enums.c.etemplate | 55 +++
gtk/nbd/nbd-enums.h.etemplate | 36 ++
gtk/nbd/nbd-export.c | 423 +++++++++++++++++++++
gtk/nbd/nbd-export.h | 65 ++++
gtk/nbd/nbd-priv.h | 84 +++++
gtk/nbd/nbd-server-session.c | 837 ++++++++++++++++++++++++++++++++++++++++++
gtk/nbd/nbd-server-session.h | 56 +++
gtk/nbd/nbd-server.c | 108 ++++++
gtk/nbd/nbd-server.h | 45 +++
gtk/nbd/nbd.c | 25 ++
gtk/nbd/nbd.h | 47 +++
14 files changed, 1829 insertions(+), 1 deletion(-)
create mode 100644 gtk/nbd/Makefile.am
create mode 100644 gtk/nbd/nbd-enums.c.etemplate
create mode 100644 gtk/nbd/nbd-enums.h.etemplate
create mode 100644 gtk/nbd/nbd-export.c
create mode 100644 gtk/nbd/nbd-export.h
create mode 100644 gtk/nbd/nbd-priv.h
create mode 100644 gtk/nbd/nbd-server-session.c
create mode 100644 gtk/nbd/nbd-server-session.h
create mode 100644 gtk/nbd/nbd-server.c
create mode 100644 gtk/nbd/nbd-server.h
create mode 100644 gtk/nbd/nbd.c
create mode 100644 gtk/nbd/nbd.h
diff --git a/configure.ac b/configure.ac
index 8ab5b6b..fc2cab4 100644
--- a/configure.ac
+++ b/configure.ac
@@ -666,6 +666,7 @@ data/spicy.desktop.in
data/spicy.nsis
po/Makefile.in
gtk/Makefile
+gtk/nbd/Makefile
gtk/controller/Makefile
doc/Makefile
doc/reference/Makefile
diff --git a/gtk/Makefile.am b/gtk/Makefile.am
index d31a396..5d29018 100644
--- a/gtk/Makefile.am
+++ b/gtk/Makefile.am
@@ -1,6 +1,6 @@
NULL =
-SUBDIRS =
+SUBDIRS = nbd
if WITH_CONTROLLER
SUBDIRS += controller
diff --git a/gtk/nbd/Makefile.am b/gtk/nbd/Makefile.am
new file mode 100644
index 0000000..50f105d
--- /dev/null
+++ b/gtk/nbd/Makefile.am
@@ -0,0 +1,46 @@
+NULL =
+
+noinst_LTLIBRARIES = libnbd.la
+
+libnbd_la_LIBADD = $(GIO_LIBS)
+# FIXME: -I.. for glib-compat atm
+libnbd_la_CPPFLAGS = \
+ -DG_LOG_DOMAIN=\"nbd\" \
+ $(GIO_CFLAGS) \
+ -I.. \
+ $(NULL)
+
+ENUMS = \
+ nbd-enums.c \
+ nbd-enums.h \
+ $(NULL)
+
+libnbd_la_SOURCES = \
+ $(ENUMS) \
+ nbd-export.c \
+ nbd-export.h \
+ nbd-priv.h \
+ nbd-server-session.c \
+ nbd-server-session.h \
+ nbd-server.c \
+ nbd-server.h \
+ nbd.c \
+ nbd.h \
+ $(NULL)
+
+ENUMS_FILES = \
+ nbd-export.h \
+ $(NULL);
+
+BUILT_SOURCES = $(ENUMS)
+
+$(ENUMS): %: %.etemplate $(ENUMS_FILES)
+ $(AM_V_GEN)glib-mkenums --template $^ > $@
+
+EXTRA_DIST = \
+ $(BUILT_SOURCES) \
+ nbd-enums.c.etemplate \
+ nbd-enums.h.etemplate \
+ $(NULL)
+
+-include $(top_srcdir)/git.mk
diff --git a/gtk/nbd/nbd-enums.c.etemplate b/gtk/nbd/nbd-enums.c.etemplate
new file mode 100644
index 0000000..d0564a3
--- /dev/null
+++ b/gtk/nbd/nbd-enums.c.etemplate
@@ -0,0 +1,55 @@
+/*** BEGIN file-header ***/
+/*
+ * Copyright (C) 2013 Marc-André Lureau <marcandre.lureau@xxxxxxxxxx>
+ *
+ * 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, see <http://www.gnu.org/licenses/>.
+ */
+
+#include "nbd-enums.h"
+
+/*** END file-header ***/
+
+/*** BEGIN file-production ***/
+#include "@filename@"
+/*** END file-production ***/
+
+
+/*** BEGIN value-header ***/
+
+GType
+@enum_name@_get_type (void)
+{
+ static volatile gsize g_define_type_id__volatile = 0;
+
+ if (g_once_init_enter (&g_define_type_id__volatile))
+ {
+ static const G@Type@Value values[] = {
+/*** END value-header ***/
+
+/*** BEGIN value-production ***/
+ { @VALUENAME@, "@VALUENAME@", "@valuenick@" },
+/*** END value-production ***/
+
+/*** BEGIN value-tail ***/
+ { 0, NULL, NULL }
+ };
+ GType g_define_type_id =
+ g_@type@_register_static (g_intern_static_string ("@EnumName@"), values);
+ g_once_init_leave (&g_define_type_id__volatile, g_define_type_id);
+ }
+
+ return g_define_type_id__volatile;
+}
+
+/*** END value-tail ***/
diff --git a/gtk/nbd/nbd-enums.h.etemplate b/gtk/nbd/nbd-enums.h.etemplate
new file mode 100644
index 0000000..ba74a14
--- /dev/null
+++ b/gtk/nbd/nbd-enums.h.etemplate
@@ -0,0 +1,36 @@
+/*** BEGIN file-header ***/
+/*
+ * Copyright (C) 2013 Marc-André Lureau <marcandre.lureau@xxxxxxxxxx>
+ *
+ * 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, see <http://www.gnu.org/licenses/>.
+ */
+
+#ifndef NBD_ENUMS_H
+#define NBD_ENUMS_H
+
+#include <glib-object.h>
+
+G_BEGIN_DECLS
+/*** END file-header ***/
+
+/*** BEGIN value-header ***/
+GType @enum_name@_get_type (void) G_GNUC_CONST;
+#define NBD_TYPE_@ENUMSHORT@ (@enum_name@_get_type ())
+/*** END value-header ***/
+
+/*** BEGIN file-tail ***/
+G_END_DECLS
+
+#endif
+/*** END file-tail ***/
diff --git a/gtk/nbd/nbd-export.c b/gtk/nbd/nbd-export.c
new file mode 100644
index 0000000..115380f
--- /dev/null
+++ b/gtk/nbd/nbd-export.c
@@ -0,0 +1,423 @@
+/* -*- Mode: C; c-basic-offset: 4; indent-tabs-mode: nil -*- */
+/*
+ Copyright (C) 2013 Red Hat, Inc.
+
+ 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.1 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, see <http://www.gnu.org/licenses/>.
+*/
+#include <errno.h>
+
+#include "nbd-export.h"
+#include "nbd-enums.h"
+#include "nbd-priv.h"
+
+struct _NbdExportClass
+{
+ GObjectClass parent_class;
+};
+
+struct _NbdExport
+{
+ GObject parent_instance;
+
+ gchar *name;
+ GFile *file;
+ GInputStream *input;
+ GOutputStream *output;
+ guint64 size;
+ NbdExportFlags flags;
+};
+
+enum {
+ PROP_0,
+
+ PROP_NAME,
+ PROP_FILE,
+ PROP_FLAGS,
+};
+static void initable_iface_init (GInitableIface *initable_iface);
+static void async_initable_iface_init (GAsyncInitableIface *async_initable_iface);
+
+G_DEFINE_TYPE_WITH_CODE(NbdExport, nbd_export, G_TYPE_OBJECT,
+ G_IMPLEMENT_INTERFACE(G_TYPE_INITABLE, initable_iface_init)
+ G_IMPLEMENT_INTERFACE(G_TYPE_ASYNC_INITABLE, async_initable_iface_init)
+ );
+
+static void
+nbd_export_init(NbdExport *self)
+{
+}
+
+static void
+nbd_export_finalize(GObject *object)
+{
+ NbdExport *self = NBD_EXPORT(object);
+
+ g_clear_object(&self->file);
+ g_free(self->name);
+
+ G_OBJECT_CLASS(nbd_export_parent_class)->finalize(object);
+}
+
+static void
+nbd_export_set_property(GObject *object, guint prop_id, const GValue *value, GParamSpec *pspec)
+{
+ g_return_if_fail(NBD_IS_EXPORT(object));
+ NbdExport *self = NBD_EXPORT(object);
+
+ switch (prop_id) {
+ case PROP_NAME:
+ self->name = g_value_dup_string(value);
+ break;
+ case PROP_FILE:
+ self->file = g_value_dup_object(value);
+ break;
+ case PROP_FLAGS:
+ self->flags = g_value_get_flags(value);
+ break;
+ default:
+ G_OBJECT_WARN_INVALID_PROPERTY_ID(object, prop_id, pspec);
+ break;
+ }
+}
+
+static void
+nbd_export_get_property(GObject *object, guint prop_id, GValue *value, GParamSpec *pspec)
+{
+ g_return_if_fail(NBD_IS_EXPORT(object));
+ NbdExport *self = NBD_EXPORT(object);
+
+ switch (prop_id) {
+ case PROP_NAME:
+ g_value_set_string(value, nbd_export_get_name(self));
+ break;
+ case PROP_FILE:
+ g_value_set_object(value, self->file);
+ break;
+ case PROP_FLAGS:
+ g_value_set_flags(value, nbd_export_get_flags(self));
+ break;
+ default:
+ G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
+ break;
+ }
+}
+
+static void
+nbd_export_class_init(NbdExportClass *klass)
+{
+ GObjectClass* object_class = G_OBJECT_CLASS (klass);
+
+ object_class->finalize = nbd_export_finalize;
+ object_class->set_property = nbd_export_set_property;
+ object_class->get_property = nbd_export_get_property;
+
+ g_object_class_install_property(object_class, PROP_NAME,
+ g_param_spec_string("name", "name", "name", NULL,
+ G_PARAM_CONSTRUCT_ONLY | G_PARAM_READWRITE |
+ G_PARAM_STATIC_STRINGS));
+
+ g_object_class_install_property(object_class, PROP_FILE,
+ g_param_spec_object("file", "file", "file", G_TYPE_FILE,
+ G_PARAM_CONSTRUCT_ONLY | G_PARAM_READWRITE |
+ G_PARAM_STATIC_STRINGS));
+
+ g_object_class_install_property(object_class, PROP_FLAGS,
+ g_param_spec_flags("flags", "flags", "flags", NBD_TYPE_EXPORT_FLAGS, 0,
+ G_PARAM_CONSTRUCT_ONLY | G_PARAM_READWRITE |
+ G_PARAM_STATIC_STRINGS));
+}
+
+static gboolean
+initable_init(GInitable *initable,
+ GCancellable *cancellable,
+ GError **error)
+{
+ NbdExport *self = NBD_EXPORT(initable);
+ GFileInfo *info;
+
+ /* the asyncinitable will call this in a thread */
+ /* finish by initable is called if init fails */
+
+ g_debug("nbd export init %d", self->flags);
+
+ if (self->flags & NBD_EXPORT_FLAGS_READWRITE) {
+ GIOStream *file;
+
+ file = G_IO_STREAM(g_file_open_readwrite(self->file, cancellable, error));
+ if (!file)
+ return FALSE;
+
+ self->output = G_OUTPUT_STREAM(g_object_ref(g_io_stream_get_output_stream(file)));
+ self->input = G_INPUT_STREAM(g_object_ref(g_io_stream_get_input_stream(file)));
+ g_object_unref(file);
+
+ if (!self->output || !self->input)
+ return FALSE;
+ } else {
+ self->input = G_INPUT_STREAM(g_file_read(self->file, cancellable, error));
+ if (!self->input)
+ return FALSE;
+ }
+
+ info = g_file_input_stream_query_info(G_FILE_INPUT_STREAM(self->input),
+ G_FILE_ATTRIBUTE_STANDARD_SIZE,
+ cancellable, error);
+ if (!info)
+ return FALSE;
+
+ self->size = g_file_info_get_attribute_uint64(info,
+ G_FILE_ATTRIBUTE_STANDARD_SIZE);
+ g_object_unref(info);
+
+ return TRUE;
+}
+
+static void
+initable_iface_init(GInitableIface *initable_iface)
+{
+ initable_iface->init = initable_init;
+}
+
+static void
+async_initable_iface_init(GAsyncInitableIface *async_initable_iface)
+{
+ /* Use default, in thread */
+}
+
+
+GFile *
+nbd_export_get_file(NbdExport *self)
+{
+ g_return_val_if_fail(NBD_IS_EXPORT(self), NULL);
+
+ return self->file;
+}
+
+const gchar *
+nbd_export_get_name(NbdExport *self)
+{
+ g_return_val_if_fail(NBD_IS_EXPORT(self), NULL);
+
+ return self->name;
+}
+
+guint64
+nbd_export_get_size(NbdExport *self)
+{
+ g_return_val_if_fail(NBD_IS_EXPORT(self), 0);
+
+ return self->size;
+}
+
+guint
+nbd_export_get_flags(NbdExport *self)
+{
+ g_return_val_if_fail(NBD_IS_EXPORT(self), 0);
+
+ return self->flags;
+}
+
+void
+nbd_export_new(gchar *name,
+ GFile *file,
+ NbdExportFlags flags,
+ GCancellable *cancellable,
+ GAsyncReadyCallback callback,
+ gpointer user_data)
+{
+ g_return_if_fail(name != NULL);
+ g_return_if_fail(G_IS_FILE(file));
+ g_return_if_fail(cancellable == NULL || G_IS_CANCELLABLE (cancellable));
+
+ g_async_initable_new_async(NBD_TYPE_EXPORT,
+ G_PRIORITY_DEFAULT,
+ cancellable,
+ callback,
+ user_data,
+ "name", name,
+ "file", file,
+ "flags", flags,
+ NULL);
+}
+
+NbdExport *
+nbd_export_new_finish(GAsyncResult *res,
+ GError **error)
+{
+ GObject *object;
+ GObject *source_object;
+
+ g_return_val_if_fail(G_IS_ASYNC_RESULT(res), NULL);
+ g_return_val_if_fail(error == NULL || *error == NULL, NULL);
+
+ source_object = g_async_result_get_source_object(res);
+ g_return_val_if_fail(source_object != NULL, NULL);
+ object = g_async_initable_new_finish(G_ASYNC_INITABLE(source_object), res, error);
+ g_object_unref(source_object);
+
+ if (object != NULL)
+ return NBD_EXPORT(object);
+
+ return NULL;
+}
+
+static gboolean
+job_ended(GIOSchedulerJob *job,
+ GSimpleAsyncResult *simple,
+ NbdRequest *req,
+ GError *error)
+{
+ g_debug("job ended %p", job);
+
+ if (error) {
+ if (req->data) {
+ g_slice_free1(req->len, req->data);
+ req->data = NULL;
+ }
+ req->error = EIO;
+ req->len = 0;
+ g_simple_async_result_take_error(simple, error);
+ } else {
+ g_simple_async_result_set_op_res_gboolean(simple, TRUE);
+ req->error = 0;
+ }
+
+ return FALSE;
+}
+
+static gboolean
+flush_job(GIOSchedulerJob *job,
+ GCancellable *cancellable,
+ gpointer user_data)
+{
+ GSimpleAsyncResult *simple = G_SIMPLE_ASYNC_RESULT(user_data);
+ NbdExport *self = NBD_EXPORT(g_async_result_get_source_object(G_ASYNC_RESULT(simple)));
+ NbdRequest *req = g_async_result_get_user_data(G_ASYNC_RESULT(simple));
+ GError *error = NULL;
+
+ g_debug("flush job %p", job);
+ g_output_stream_flush(self->output, cancellable, &error);
+ g_object_unref(self);
+
+ return job_ended(job, simple, req, error);
+}
+
+static gboolean
+read_job(GIOSchedulerJob *job,
+ GCancellable *cancellable,
+ gpointer user_data)
+{
+ GSimpleAsyncResult *simple = G_SIMPLE_ASYNC_RESULT(user_data);
+ NbdExport *self = NBD_EXPORT(g_async_result_get_source_object(G_ASYNC_RESULT(simple)));
+ NbdRequest *req = g_async_result_get_user_data(G_ASYNC_RESULT(simple));
+ GError *error = NULL;
+ gssize bytes_read;
+
+ g_debug("read job %p", job);
+
+ if (!g_seekable_seek(G_SEEKABLE(self->input), req->from, G_SEEK_SET, cancellable, &error))
+ goto end;
+
+ g_warn_if_fail(req->data == NULL);
+ req->data = g_slice_alloc(req->len);
+
+ if (!g_input_stream_read_all(G_INPUT_STREAM(self->input),
+ req->data, req->len, &bytes_read, cancellable, &error))
+ goto end;
+
+end:
+ g_object_unref(self);
+
+ return job_ended(job, simple, req, error);
+}
+
+static void
+job_notify(gpointer data)
+{
+ GSimpleAsyncResult *simple = data;
+
+ g_simple_async_result_complete_in_idle(simple);
+ g_object_unref(simple);
+}
+
+
+G_GNUC_INTERNAL void
+nbd_export_request(NbdExport *self,
+ NbdRequest *request,
+ GCancellable *cancellable,
+ GAsyncReadyCallback callback,
+ gpointer data)
+{
+ GSimpleAsyncResult *simple;
+
+ g_return_if_fail(NBD_IS_EXPORT(self));
+ g_return_if_fail(G_IS_FILE_INPUT_STREAM(self->input));
+ g_return_if_fail(cancellable == NULL || G_IS_CANCELLABLE (cancellable));
+
+ guint32 command = request->type & NBD_CMD_MASK_COMMAND;
+ g_debug("NBD export request %u", command);
+
+ simple = g_simple_async_result_new(G_OBJECT(self),
+ callback,
+ request,
+ nbd_export_request);
+
+ if (!(self->flags & NBD_EXPORT_FLAGS_READWRITE) &&
+ command != NBD_CMD_READ)
+ goto unhandled;
+
+ switch (command) {
+ case NBD_CMD_READ:
+ g_io_scheduler_push_job(read_job, simple, job_notify,
+ G_PRIORITY_DEFAULT, cancellable);
+ break;
+ case NBD_CMD_FLUSH:
+ g_io_scheduler_push_job(flush_job, simple, job_notify,
+ G_PRIORITY_DEFAULT, cancellable);
+ break;
+ case NBD_CMD_WRITE:
+ case NBD_CMD_TRIM:
+ g_debug("Write commands not yet supported");
+ default:
+ unhandled:
+ request->error = EINVAL;
+ g_simple_async_result_set_error(simple,
+ NBD_ERROR,
+ NBD_ERROR_FAILED,
+ "unhandled NBD request %u", command);
+ job_notify(simple);
+ }
+}
+
+G_GNUC_INTERNAL gboolean
+nbd_export_request_finish(NbdExport *self,
+ GAsyncResult *result,
+ GError **error)
+{
+ GSimpleAsyncResult *simple;
+
+ g_return_val_if_fail(NBD_IS_EXPORT(self), FALSE);
+ g_return_val_if_fail(error == NULL || *error == NULL, FALSE);
+ g_return_val_if_fail(g_simple_async_result_is_valid(result,
+ G_OBJECT(self),
+ nbd_export_request),
+ FALSE);
+
+ simple = (GSimpleAsyncResult *)result;
+
+ if (g_simple_async_result_propagate_error(simple, error))
+ return FALSE;
+
+ return g_simple_async_result_get_op_res_gboolean(simple);
+}
diff --git a/gtk/nbd/nbd-export.h b/gtk/nbd/nbd-export.h
new file mode 100644
index 0000000..f1b4c35
--- /dev/null
+++ b/gtk/nbd/nbd-export.h
@@ -0,0 +1,65 @@
+/* -*- Mode: C; c-basic-offset: 4; indent-tabs-mode: nil -*- */
+/*
+ Copyright (C) 2013 Red Hat, Inc.
+
+ 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.1 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, see <http://www.gnu.org/licenses/>.
+*/
+
+#ifndef _NBD_EXPORT_H_
+# define _NBD_EXPORT_H_
+
+#include <gio/gio.h>
+
+G_BEGIN_DECLS
+
+/**
+ * NbdExportFlags:
+ *
+ **/
+typedef enum
+{
+ NBD_EXPORT_FLAGS_NONE = 0,
+
+ NBD_EXPORT_FLAGS_READWRITE = 1 << 0,
+} NbdExportFlags;
+
+#define NBD_TYPE_EXPORT (nbd_export_get_type ())
+#define NBD_EXPORT(export) (G_TYPE_CHECK_INSTANCE_CAST ((export), NBD_TYPE_EXPORT, NbdExport))
+#define NBD_EXPORT_CLASS(klass) (G_TYPE_CHECK_CLASS_CAST ((klass), NBD_TYPE_EXPORT, NbdExportClass))
+#define NBD_IS_EXPORT(export) (G_TYPE_CHECK_INSTANCE_TYPE ((export), NBD_TYPE_EXPORT))
+#define NBD_IS_EXPORT_CLASS(klass) (G_TYPE_CHECK_CLASS_TYPE ((klass), NBD_TYPE_EXPORT))
+#define NBD_EXPORT_GET_CLASS(export) (G_TYPE_INSTANCE_GET_CLASS ((export), NBD_TYPE_EXPORT, NbdExportClass))
+
+/* TODO: we could have a base class with overriable op */
+
+typedef struct _NbdExportClass NbdExportClass;
+typedef struct _NbdExport NbdExport;
+
+GType nbd_export_get_type (void) G_GNUC_CONST;
+void nbd_export_new (gchar *name,
+ GFile *file,
+ NbdExportFlags flags,
+ GCancellable *cancellable,
+ GAsyncReadyCallback callback,
+ gpointer user_data);
+NbdExport* nbd_export_new_finish (GAsyncResult *res,
+ GError **error);
+GFile* nbd_export_get_file (NbdExport *export);
+const gchar* nbd_export_get_name (NbdExport *export);
+guint64 nbd_export_get_size (NbdExport *export);
+guint nbd_export_get_flags (NbdExport *export);
+
+G_END_DECLS
+
+#endif /* _NBD_EXPORT_H_ */
diff --git a/gtk/nbd/nbd-priv.h b/gtk/nbd/nbd-priv.h
new file mode 100644
index 0000000..8c4712e
--- /dev/null
+++ b/gtk/nbd/nbd-priv.h
@@ -0,0 +1,84 @@
+/* -*- Mode: C; c-basic-offset: 4; indent-tabs-mode: nil -*- */
+/*
+ Copyright (C) 2013 Red Hat, Inc.
+
+ 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.1 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, see <http://www.gnu.org/licenses/>.
+*/
+
+#ifndef _NBD_PRIV_H_
+# define _NBD_PRIV_H_
+
+#include "nbd.h"
+#include "glib-compat.h"
+
+/* This is all part of the "official" NBD API */
+
+#define NBD_REQUEST_SIZE (4 + 4 + 8 + 8 + 4)
+#define NBD_REPLY_SIZE (4 + 4 + 8)
+#define NBD_REQUEST_MAGIC 0x25609513
+#define NBD_REPLY_MAGIC 0x67446698
+#define NBD_OPTS_MAGIC 0x49484156454F5054LL
+#define NBD_CLIENT_MAGIC 0x0000420281861253LL
+
+#define NBD_OPT_EXPORT_NAME (1 << 0)
+
+#define NBD_FLAG_HAS_FLAGS (1 << 0) /* Flags are there */
+#define NBD_FLAG_READ_ONLY (1 << 1) /* Device is read-only */
+#define NBD_FLAG_SEND_FLUSH (1 << 2) /* Send FLUSH */
+#define NBD_FLAG_SEND_FUA (1 << 3) /* Send FUA (Force Unit Access) */
+#define NBD_FLAG_ROTATIONAL (1 << 4) /* Use elevator algorithm - rotational media */
+#define NBD_FLAG_SEND_TRIM (1 << 5) /* Send TRIM (discard) */
+
+#define NBD_CMD_MASK_COMMAND 0x0000ffff
+#define NBD_CMD_FLAG_FUA (1 << 16)
+
+enum {
+ NBD_CMD_READ = 0,
+ NBD_CMD_WRITE = 1,
+ NBD_CMD_DISC = 2,
+ NBD_CMD_FLUSH = 3,
+ NBD_CMD_TRIM = 4
+};
+
+#define NBD_DEFAULT_PORT 10809
+
+/* Maximum size of a single READ/WRITE data buffer */
+#define NBD_MAX_BUFFER_SIZE (32 * 1024 * 1024)
+
+/* FIXME: the export backend might support more but currently GIO
+ * doesn't allow to do multiple outstanding IO */
+#define NBD_MAX_REQUESTS 1
+
+typedef struct NbdRequest
+{
+ guint32 type;
+ guint64 handle;
+ guint64 from;
+ guint32 len;
+ guint8 *data;
+ NbdServerSession *session;
+ guint32 error;
+} NbdRequest;
+
+void nbd_export_request (NbdExport *export,
+ NbdRequest *request,
+ GCancellable *cancellable,
+ GAsyncReadyCallback cb,
+ gpointer data);
+
+gboolean nbd_export_request_finish (NbdExport *export,
+ GAsyncResult *res,
+ GError **error);
+
+#endif /* _NBD_PRIV_H_ */
diff --git a/gtk/nbd/nbd-server-session.c b/gtk/nbd/nbd-server-session.c
new file mode 100644
index 0000000..0b5b175
--- /dev/null
+++ b/gtk/nbd/nbd-server-session.c
@@ -0,0 +1,837 @@
+/* -*- Mode: C; c-basic-offset: 4; indent-tabs-mode: nil -*- */
+/*
+ Copyright (C) 2013 Red Hat, Inc.
+
+ 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.1 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, see <http://www.gnu.org/licenses/>.
+*/
+
+#include "nbd-server-session.h"
+#include "nbd-priv.h"
+
+struct _NbdServerSessionClass
+{
+ GObjectClass parent_class;
+};
+
+struct _NbdServerSession
+{
+ GObject parent_instance;
+ GQueue *requests;
+
+ NbdServer *server;
+ NbdExport *export;
+ GIOStream *stream;
+
+ GDataOutputStream *dout;
+ GDataInputStream *din;
+
+ NbdRequest *flushing; /* weak */
+ gboolean pending;
+ GSimpleAsyncResult *closing;
+ gulong closing_id;
+ GCancellable *cancellable;
+ GCancellable *closing_cancellable;
+
+ gsize name_len;
+ gchar name[256];
+};
+
+enum {
+ PROP_0,
+ PROP_SERVER,
+ PROP_EXPORT,
+ PROP_STREAM,
+ PROP_CLOSED,
+};
+
+enum {
+ SIGNAL_LAST,
+};
+
+static guint signals[SIGNAL_LAST];
+
+static void async_initable_iface_init(GAsyncInitableIface *iface);
+static void export_request_finished(GObject *source_object,
+ GAsyncResult *res,
+ gpointer user_data);
+
+G_DEFINE_TYPE_WITH_CODE(NbdServerSession, nbd_server_session, G_TYPE_OBJECT,
+ G_IMPLEMENT_INTERFACE(G_TYPE_ASYNC_INITABLE, async_initable_iface_init)
+ );
+
+static void
+nbd_server_session_init(NbdServerSession *self)
+{
+ self->requests = g_queue_new();
+}
+
+static void
+nbd_server_session_finalize(GObject *object)
+{
+ NbdServerSession *self = NBD_SERVER_SESSION(object);
+
+ g_debug("NBD session finalize: %d", self->pending);
+
+ g_queue_free_full(self->requests, NULL); /* FIXME */
+
+ /* the async must be finished before, or it will hold a ref */
+ g_warn_if_fail(self->closing == NULL);
+
+ g_clear_object(&self->server);
+ g_clear_object(&self->export);
+ g_clear_object(&self->stream);
+ g_clear_object(&self->dout);
+ g_clear_object(&self->din);
+ g_clear_object(&self->cancellable);
+
+ G_OBJECT_CLASS(nbd_server_session_parent_class)->finalize(object);
+}
+
+static void
+nbd_server_session_set_property(GObject *object, guint prop_id, const GValue *value, GParamSpec *pspec)
+{
+ g_return_if_fail(NBD_IS_SERVER_SESSION(object));
+ NbdServerSession *self = NBD_SERVER_SESSION(object);
+
+ switch (prop_id) {
+ case PROP_SERVER:
+ self->server = g_value_dup_object(value);
+ break;
+ case PROP_EXPORT:
+ self->export = g_value_dup_object(value);
+ break;
+ case PROP_STREAM:
+ self->stream = g_value_dup_object(value);
+ break;
+ default:
+ G_OBJECT_WARN_INVALID_PROPERTY_ID(object, prop_id, pspec);
+ break;
+ }
+}
+
+static gboolean
+nbd_server_session_get_closed(NbdServerSession *self)
+{
+ g_return_val_if_fail(NBD_IS_SERVER_SESSION(self), FALSE);
+
+ return !self->stream || g_io_stream_is_closed(self->stream);
+}
+
+
+
+static void
+nbd_server_session_get_property(GObject *object, guint prop_id, GValue *value, GParamSpec *pspec)
+{
+ g_return_if_fail(NBD_IS_SERVER_SESSION(object));
+ NbdServerSession *self = NBD_SERVER_SESSION(object);
+
+ switch (prop_id) {
+ case PROP_SERVER:
+ g_value_set_object(value, self->server);
+ break;
+ case PROP_EXPORT:
+ g_value_set_object(value, self->export);
+ break;
+ case PROP_STREAM:
+ g_value_set_object(value, self->stream);
+ break;
+ case PROP_CLOSED:
+ g_value_set_boolean(value, nbd_server_session_get_closed(self));
+ break;
+ default:
+ G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
+ break;
+ }
+}
+
+static void
+nbd_server_session_class_init(NbdServerSessionClass *klass)
+{
+ GObjectClass* object_class = G_OBJECT_CLASS (klass);
+
+ object_class->finalize = nbd_server_session_finalize;
+ object_class->set_property = nbd_server_session_set_property;
+ object_class->get_property = nbd_server_session_get_property;
+
+ g_object_class_install_property(object_class, PROP_SERVER,
+ g_param_spec_object("server", "server", "server", NBD_TYPE_SERVER,
+ G_PARAM_CONSTRUCT_ONLY | G_PARAM_READWRITE |
+ G_PARAM_STATIC_STRINGS));
+
+ g_object_class_install_property(object_class, PROP_EXPORT,
+ g_param_spec_object("export", "export", "export", NBD_TYPE_EXPORT,
+ G_PARAM_CONSTRUCT_ONLY | G_PARAM_READWRITE |
+ G_PARAM_STATIC_STRINGS));
+
+ g_object_class_install_property(object_class, PROP_STREAM,
+ g_param_spec_object("stream", "stream", "stream", G_TYPE_IO_STREAM,
+ G_PARAM_CONSTRUCT_ONLY | G_PARAM_READWRITE |
+ G_PARAM_STATIC_STRINGS));
+
+ g_object_class_install_property(object_class, PROP_CLOSED,
+ g_param_spec_boolean("closed", "closed", "closed", FALSE,
+ G_PARAM_READABLE |
+ G_PARAM_STATIC_STRINGS));
+}
+
+void
+nbd_server_session_new(NbdServer *server,
+ GIOStream *stream,
+ NbdExport *export,
+ GCancellable *cancellable,
+ GAsyncReadyCallback callback,
+ gpointer user_data)
+{
+ g_return_if_fail(NBD_IS_SERVER(server));
+ g_return_if_fail(G_IS_IO_STREAM(stream));
+ g_return_if_fail(!export || NBD_IS_EXPORT(export));
+ g_return_if_fail(cancellable == NULL || G_IS_CANCELLABLE (cancellable));
+
+ g_async_initable_new_async(NBD_TYPE_SERVER_SESSION,
+ G_PRIORITY_DEFAULT,
+ cancellable,
+ callback,
+ user_data,
+ "server", server,
+ "stream", stream,
+ "export", export,
+ NULL);
+}
+
+NbdServerSession*
+nbd_server_session_new_finish(GAsyncResult *res,
+ GError **error)
+{
+ GObject *object;
+ GObject *source_object;
+
+ g_return_val_if_fail(G_IS_ASYNC_RESULT(res), NULL);
+ g_return_val_if_fail(error == NULL || *error == NULL, NULL);
+
+ source_object = g_async_result_get_source_object(res);
+ g_return_val_if_fail(source_object != NULL, NULL);
+ object = g_async_initable_new_finish(G_ASYNC_INITABLE(source_object), res, error);
+ g_object_unref(source_object);
+
+ if (object != NULL)
+ return NBD_SERVER_SESSION(object);
+
+ return NULL;
+}
+
+static gboolean
+nbd_server_session_has_pending(NbdServerSession *self)
+{
+ /* the 1st request is waiting */
+ return self->pending || g_queue_get_length(self->requests) > 1;
+}
+
+#define RETURN_ERROR(simple, error) G_STMT_START{ \
+ g_simple_async_result_take_error(simple, error); \
+ g_simple_async_result_complete(simple); \
+ g_object_unref(simple); \
+ return; \
+ }G_STMT_END
+
+static void
+session_close(NbdServerSession *self, GError *error)
+{
+ g_debug("session close, pending:%d",
+ nbd_server_session_has_pending(self));
+
+ if (error) {
+ g_warning("NBD error: %s", error->message);
+ g_clear_error(&error);
+ }
+
+ /* we don't close the stream ourself,
+ leave that to the last unref handler */
+ if (self->stream) {
+ g_clear_object(&self->stream);
+ g_object_notify(G_OBJECT(self), "closed");
+ }
+
+ /* this will cancel the waiting request, if any */
+ if (!nbd_server_session_has_pending(self))
+ g_cancellable_cancel(self->cancellable);
+
+ /* finish the async, if any */
+ if (self->closing &&
+ g_queue_get_length(self->requests) == 0) {
+
+ g_simple_async_result_set_op_res_gboolean(self->closing, TRUE);
+ g_simple_async_result_complete_in_idle(self->closing);
+ g_clear_object(&self->closing);
+
+ g_cancellable_disconnect(self->closing_cancellable, self->closing_id);
+ self->closing_id = 0;
+ }
+}
+
+static void
+close_cancelled_handler(GCancellable *cancellable,
+ gpointer user_data)
+{
+ NbdServerSession *self = user_data;
+
+ /* if closing is forced, cancel all pending requests */
+ g_cancellable_cancel(self->cancellable);
+}
+
+void
+nbd_server_session_close_async(NbdServerSession *self,
+ GCancellable *cancellable,
+ GAsyncReadyCallback callback,
+ gpointer user_data)
+
+{
+ g_return_if_fail(NBD_IS_SERVER_SESSION(self));
+ g_return_if_fail(!self->closing);
+
+ if (cancellable) {
+ self->closing_cancellable = cancellable;
+ self->closing_id = g_cancellable_connect(cancellable,
+ G_CALLBACK(close_cancelled_handler),
+ self, NULL);
+ }
+
+ self->closing =
+ g_simple_async_result_new(G_OBJECT(self),
+ callback,
+ user_data,
+ nbd_server_session_close_async);
+
+ session_close(self, NULL);
+}
+
+gboolean
+nbd_server_session_close_finish(NbdServerSession *self,
+ GAsyncResult *result,
+ GError **error)
+{
+ GSimpleAsyncResult *simple;
+
+ g_return_val_if_fail(NBD_IS_SERVER_SESSION(self), FALSE);
+ g_return_val_if_fail(error == NULL || *error == NULL, FALSE);
+ g_return_val_if_fail(g_simple_async_result_is_valid(result,
+ G_OBJECT(self),
+ nbd_server_session_close_async),
+ FALSE);
+
+ simple = (GSimpleAsyncResult *)result;
+ self->closing = NULL;
+
+ if (g_simple_async_result_propagate_error(simple, error))
+ return FALSE;
+
+ return g_simple_async_result_get_op_res_gboolean(simple);
+}
+
+static void receive_request(NbdServerSession *self);
+
+static NbdRequest *
+nbd_request_get(NbdServerSession *self)
+{
+ NbdRequest *req;
+
+ g_debug("Get NBD request, nb: %d flushing: %p",
+ g_queue_get_length(self->requests), self->flushing);
+
+ if (!self->stream ||
+ g_queue_get_length(self->requests) >= NBD_MAX_REQUESTS ||
+ self->flushing)
+ return NULL;
+
+ req = g_slice_new0(NbdRequest);
+ req->session = self;
+ g_queue_push_head(self->requests, req);
+
+ return req;
+}
+
+static void
+nbd_request_put(NbdServerSession *self, NbdRequest *req)
+{
+ g_warn_if_fail(g_queue_remove(self->requests, req));
+
+ if (self->flushing &&
+ g_queue_get_length(self->requests) == 1) {
+ NbdRequest *req = self->flushing;
+
+ g_debug("flushed all pending requests");
+ self->flushing = NULL;
+ nbd_export_request(self->export, req,
+ self->cancellable, export_request_finished,
+ req);
+ }
+
+ /* that could be useful? */
+ /* if (g_queue_get_length(self->requests) == 0) */
+ /* g_object_notify(G_OBJECT(self), "pending"); */
+
+ /* loop can be interrupted by request_get() == NULL */
+ receive_request(self);
+}
+
+static void
+reply_flushed(GObject *source_object,
+ GAsyncResult *res,
+ gpointer user_data)
+{
+ GError *error = NULL;
+ NbdRequest *req = user_data;
+ NbdServerSession *self = req->session;
+
+ g_debug("NBD reply flushed");
+ if (!g_output_stream_flush_finish(G_OUTPUT_STREAM(source_object), res, &error))
+ session_close(self, error);
+
+ nbd_request_put(self, req);
+}
+
+static void
+nbd_send_reply(NbdRequest *req)
+{
+ NbdServerSession *self = req->session;
+ GError *error = NULL;
+
+ g_warn_if_fail(!req->len || req->data);
+
+ g_debug("Send reply: "
+ "{ .error = %u, .len = %u }", req->error, req->len);
+
+ if (!g_data_output_stream_put_uint32(self->dout,
+ NBD_REPLY_MAGIC, self->cancellable, &error) ||
+ !g_data_output_stream_put_uint32(self->dout,
+ req->error, self->cancellable, &error) ||
+ !g_data_output_stream_put_uint64(self->dout,
+ req->handle, self->cancellable, &error))
+ goto err;
+
+ if (req->len && req->data &&
+ g_output_stream_write(G_OUTPUT_STREAM(self->dout),
+ req->data, req->len, self->cancellable, &error) < 0)
+ goto err;
+
+ g_output_stream_flush_async(G_OUTPUT_STREAM(self->dout),
+ G_PRIORITY_DEFAULT, self->cancellable,
+ reply_flushed, req);
+ return;
+
+ err:
+ session_close(self, error);
+ nbd_request_put(self, req);
+}
+
+static void
+export_request_finished(GObject *source_object,
+ GAsyncResult *res,
+ gpointer user_data)
+{
+ GError *error = NULL;
+ NbdRequest *req = user_data;
+ NbdServerSession *self = req->session;
+
+ if (!nbd_export_request_finish(self->export, res, &error)) {
+ g_warning("request error: %s", error->message);
+ g_clear_error(&error);
+ }
+
+ nbd_send_reply(req);
+}
+
+static void
+handle_request(NbdServerSession *self, NbdRequest *req)
+{
+ guint command = req->type & NBD_CMD_MASK_COMMAND;
+
+ switch (command) {
+ case NBD_CMD_DISC:
+ session_close(self, NULL);
+ break;
+ case NBD_CMD_FLUSH:
+ /* wait until all previous requests are done */
+ g_warn_if_fail(!self->flushing);
+ self->flushing = req;
+ break;
+ default:
+ nbd_export_request(self->export, req,
+ self->cancellable,
+ export_request_finished,
+ req);
+ }
+}
+
+static void
+write_filled(GObject *source_object,
+ GAsyncResult *res,
+ gpointer user_data)
+{
+ GError *error = NULL;
+ NbdRequest *req = user_data;
+ NbdServerSession *self = req->session;
+
+ if (g_input_stream_read_finish(G_INPUT_STREAM(self->din), res, &error) < req->len) {
+ session_close(self, error);
+ nbd_request_put(self, req);
+ return;
+ }
+
+ handle_request(self, req);
+}
+
+static void
+request_filled(GObject *source_object,
+ GAsyncResult *res,
+ gpointer user_data)
+{
+ NbdRequest *req = user_data;
+ NbdServerSession *self = req->session;
+ GError *error = NULL;
+ guint32 magic, command;
+
+ if (g_buffered_input_stream_fill_finish(G_BUFFERED_INPUT_STREAM(self->din), res, &error) != NBD_REQUEST_SIZE)
+ goto end;
+
+ g_return_if_fail(g_buffered_input_stream_get_available(G_BUFFERED_INPUT_STREAM(self->din)) >= NBD_REQUEST_SIZE);
+
+ magic = g_data_input_stream_read_uint32(self->din, self->cancellable, &error);
+ req->type = g_data_input_stream_read_uint32(self->din, self->cancellable, &error);
+ req->handle = g_data_input_stream_read_uint64(self->din, self->cancellable, &error);
+ req->from = g_data_input_stream_read_uint64(self->din, self->cancellable, &error);
+ req->len = g_data_input_stream_read_uint32(self->din, self->cancellable, &error);
+
+ g_debug("Got request: "
+ "{ magic = 0x%x, .type = %u, from = %" G_GINT64_FORMAT " , len = %u }",
+ magic, req->type, req->from, req->len);
+
+ if (magic != NBD_REQUEST_MAGIC)
+ error = g_error_new(NBD_ERROR, NBD_ERROR_FAILED, "Invalid request magic");
+ else if (req->len > NBD_MAX_BUFFER_SIZE)
+ error = g_error_new(NBD_ERROR, NBD_ERROR_FAILED, "len (%u) is larger than max len (%u)", req->len, NBD_MAX_BUFFER_SIZE);
+ else if ((req->from + req->len) < req->from)
+ error = g_error_new(NBD_ERROR, NBD_ERROR_FAILED, "Integer overflow");
+ else if ((req->from + req->len) > nbd_export_get_size(self->export))
+ error = g_error_new(NBD_ERROR, NBD_ERROR_FAILED, "Operation past EOF");
+
+ if (error)
+ goto end;
+
+ command = req->type & NBD_CMD_MASK_COMMAND;
+ if (command == NBD_CMD_WRITE) {
+ req->data = g_slice_alloc(req->len);
+ g_input_stream_read_async(G_INPUT_STREAM(self->din),
+ req->data, req->len, G_PRIORITY_DEFAULT,
+ self->cancellable, write_filled, req);
+ } else
+ handle_request(self, req);
+
+ end:
+ if (error) {
+ if (error->code == G_IO_ERROR_CANCELLED &&
+ self->closing) {
+ g_debug("request cancelled");
+ g_clear_error(&error);
+ }
+ nbd_request_put(self, req);
+ session_close(self, error);
+ return;
+ }
+
+ /* will receive concurrent requests up to MAX REQUEST */
+ receive_request(self);
+}
+
+static void
+receive_request(NbdServerSession *self)
+{
+ g_return_if_fail(NBD_IS_SERVER_SESSION(self));
+
+ NbdRequest *req = nbd_request_get(self);
+ if (!req) {
+ g_debug("Request processing is interrupted");
+ return;
+ }
+
+ g_return_if_fail(!g_input_stream_has_pending(G_INPUT_STREAM(self->din)));
+ g_buffered_input_stream_fill_async(G_BUFFERED_INPUT_STREAM(self->din),
+ NBD_REQUEST_SIZE, G_PRIORITY_DEFAULT,
+ self->cancellable, request_filled, req);
+}
+
+static void
+negotiate_ended(GObject *source_object,
+ GAsyncResult *res,
+ gpointer user_data)
+{
+ GSimpleAsyncResult *simple = user_data;
+
+ g_simple_async_result_set_op_res_gboolean(simple, TRUE);
+ g_simple_async_result_complete(simple);
+}
+
+static void
+size_and_flags(NbdServerSession *self,
+ GSimpleAsyncResult *simple,
+ gboolean with_server_flags)
+{
+ GError *error = NULL;
+ gchar reserved[124] = { 0, };
+ const int myflags = (NBD_FLAG_HAS_FLAGS | NBD_FLAG_SEND_TRIM |
+ NBD_FLAG_SEND_FLUSH | NBD_FLAG_SEND_FUA);
+
+ if (!g_data_output_stream_put_uint64(self->dout,
+ nbd_export_get_size(self->export), self->cancellable, &error) ||
+ (with_server_flags &&
+ !g_data_output_stream_put_uint16(self->dout,
+ 0, self->cancellable, &error)) ||
+ !g_data_output_stream_put_uint16(self->dout,
+ nbd_export_get_flags(self->export) | myflags, self->cancellable, &error) ||
+ g_output_stream_write(G_OUTPUT_STREAM(self->dout),
+ reserved, sizeof(reserved), self->cancellable, &error) == -1)
+ RETURN_ERROR(simple, error);
+
+ g_output_stream_flush_async(G_OUTPUT_STREAM(self->dout),
+ G_PRIORITY_DEFAULT, self->cancellable,
+ negotiate_ended, simple);
+}
+
+static void
+header_named(GObject *source_object,
+ GAsyncResult *res,
+ gpointer user_data)
+{
+ GError *error = NULL;
+ NbdExport *export;
+ GSimpleAsyncResult *simple = user_data;
+ NbdServerSession *self = NBD_SERVER_SESSION(g_async_result_get_source_object(G_ASYNC_RESULT(simple)));
+
+ g_return_if_fail(g_buffered_input_stream_get_available(G_BUFFERED_INPUT_STREAM(self->din)) >= self->name_len);
+
+ if (g_input_stream_read(G_INPUT_STREAM(self->din), self->name, self->name_len, self->cancellable, &error) < self->name_len)
+ goto err;
+
+ self->name[self->name_len] = '\0';
+ g_debug("Request export name: %s", self->name);
+
+ export = nbd_server_get_export(self->server, self->name);
+ if (!export) {
+ g_simple_async_result_set_error(simple,
+ NBD_ERROR,
+ NBD_ERROR_FAILED,
+ "Couldn't find export");
+ goto complete;
+ }
+
+ g_warn_if_fail(self->export == NULL);
+ self->export = g_object_ref(export);
+
+ size_and_flags(self, simple, FALSE);
+ return;
+
+err:
+ if (error != NULL)
+ g_simple_async_result_take_error(simple, error);
+ else
+ g_simple_async_result_set_error(simple,
+ NBD_ERROR,
+ NBD_ERROR_FAILED,
+ "Invalid header");
+complete:
+ g_object_unref(self);
+ g_simple_async_result_complete(simple);
+ g_object_unref(simple);
+}
+
+#define HEADER_SIZE (3 * sizeof(guint32) + sizeof(guint64))
+
+static void
+header_filled(GObject *source_object,
+ GAsyncResult *res,
+ gpointer user_data)
+{
+ GError *error = NULL;
+ GSimpleAsyncResult *simple = user_data;
+ NbdServerSession *self = NBD_SERVER_SESSION(g_async_result_get_source_object(G_ASYNC_RESULT(simple)));
+
+ if (g_buffered_input_stream_fill_finish(G_BUFFERED_INPUT_STREAM(self->din), res, &error) != HEADER_SIZE)
+ goto err;
+
+ g_return_if_fail(g_buffered_input_stream_get_available(G_BUFFERED_INPUT_STREAM(self->din)) >= HEADER_SIZE);
+
+ if (g_data_input_stream_read_uint32(self->din, self->cancellable, &error) != 0 ||
+ g_data_input_stream_read_uint64(self->din, self->cancellable, &error) != NBD_OPTS_MAGIC ||
+ g_data_input_stream_read_uint32(self->din, self->cancellable, &error) != NBD_OPT_EXPORT_NAME)
+ goto err;
+
+ self->name_len = g_data_input_stream_read_uint32(self->din, self->cancellable, &error);
+ g_debug("getting name len: %" G_GSIZE_FORMAT, self->name_len);
+ if (error || self->name_len >= sizeof(self->name))
+ goto err;
+
+ g_buffered_input_stream_fill_async(G_BUFFERED_INPUT_STREAM(self->din),
+ self->name_len, G_PRIORITY_DEFAULT,
+ self->cancellable, header_named, simple);
+ return;
+
+ err:
+ if (error != NULL)
+ g_simple_async_result_take_error(simple, error);
+ else
+ g_simple_async_result_set_error(simple,
+ NBD_ERROR,
+ NBD_ERROR_FAILED,
+ "Invalid header");
+
+ g_object_unref(self);
+ g_simple_async_result_complete(simple);
+ g_object_unref(simple);
+}
+
+static void
+negotiate_flushed(GObject *source_object,
+ GAsyncResult *res,
+ gpointer user_data)
+{
+ GError *error = NULL;
+ GSimpleAsyncResult *simple = user_data;
+
+ if (!g_output_stream_flush_finish(G_OUTPUT_STREAM(source_object), res, &error))
+ RETURN_ERROR(simple, error);
+
+ NbdServerSession *self = NBD_SERVER_SESSION(g_async_result_get_source_object(G_ASYNC_RESULT(simple)));
+
+ g_buffered_input_stream_fill_async(G_BUFFERED_INPUT_STREAM(self->din),
+ HEADER_SIZE, G_PRIORITY_DEFAULT,
+ self->cancellable, header_filled, simple);
+
+ g_object_unref(self);
+}
+
+static void
+nbd_server_session_negotiate_async(NbdServerSession *self,
+ GCancellable *cancellable,
+ GAsyncReadyCallback callback,
+ gpointer user_data)
+{
+ GError *error = NULL;
+ GSimpleAsyncResult *simple;
+
+ g_return_if_fail(NBD_IS_SERVER_SESSION(self));
+ g_return_if_fail(cancellable == NULL || G_IS_CANCELLABLE (cancellable));
+ g_return_if_fail(!nbd_server_session_has_pending(self));
+
+ g_debug("NBD negotiate %p", self);
+
+ self->pending = TRUE;
+ self->cancellable = g_object_ref(cancellable);
+ simple = g_simple_async_result_new(G_OBJECT(self),
+ callback,
+ user_data,
+ nbd_server_session_negotiate_async);
+
+ if (!g_data_output_stream_put_string(self->dout,
+ "NBDMAGIC", cancellable, &error))
+ RETURN_ERROR(simple, error);
+
+ if (self->export) {
+ if (!g_data_output_stream_put_uint64(self->dout,
+ NBD_CLIENT_MAGIC, cancellable, &error))
+ RETURN_ERROR(simple, error);
+
+ size_and_flags(self, simple, TRUE);
+ } else {
+ if (!g_data_output_stream_put_uint64(self->dout,
+ NBD_OPTS_MAGIC, cancellable, &error) ||
+ !g_data_output_stream_put_uint16(self->dout,
+ /* server flags */
+ 0, cancellable, &error))
+ RETURN_ERROR(simple, error);
+
+ g_output_stream_flush_async(G_OUTPUT_STREAM(self->dout),
+ G_PRIORITY_DEFAULT, cancellable,
+ negotiate_flushed, simple);
+ }
+
+}
+
+static gboolean
+nbd_server_session_negotiate_finish(NbdServerSession *self,
+ GAsyncResult *result,
+ GError **error)
+{
+ GSimpleAsyncResult *simple;
+
+ g_return_val_if_fail(NBD_IS_SERVER_SESSION(self), FALSE);
+ g_return_val_if_fail(error == NULL || *error == NULL, FALSE);
+ g_return_val_if_fail(g_simple_async_result_is_valid(result,
+ G_OBJECT(self),
+ nbd_server_session_negotiate_async),
+ FALSE);
+
+ self->pending = FALSE;
+ g_clear_object(&self->cancellable);
+
+ simple = (GSimpleAsyncResult *)result;
+
+ if (g_simple_async_result_propagate_error(simple, error))
+ return FALSE;
+
+ self->cancellable = g_cancellable_new();
+ receive_request(self);
+
+ return g_simple_async_result_get_op_res_gboolean(simple);
+}
+
+static void
+init_async(GAsyncInitable *initable,
+ int io_priority,
+ GCancellable *cancellable,
+ GAsyncReadyCallback callback,
+ gpointer user_data)
+{
+ NbdServerSession *self = NBD_SERVER_SESSION(initable);
+
+ GInputStream *input = g_io_stream_get_input_stream(self->stream);
+ self->din = g_data_input_stream_new(input);
+ g_data_input_stream_set_byte_order(self->din,
+ G_DATA_STREAM_BYTE_ORDER_BIG_ENDIAN);
+
+ GOutputStream *output = g_io_stream_get_output_stream(self->stream);
+ GOutputStream *buffered = g_buffered_output_stream_new(output);
+ g_buffered_output_stream_set_auto_grow(G_BUFFERED_OUTPUT_STREAM(buffered), TRUE);
+ self->dout = g_data_output_stream_new(buffered);
+ g_object_unref(buffered);
+ g_data_output_stream_set_byte_order(self->dout,
+ G_DATA_STREAM_BYTE_ORDER_BIG_ENDIAN);
+
+ nbd_server_session_negotiate_async(self, cancellable, callback, user_data);
+}
+
+static gboolean
+init_finish(GAsyncInitable *initable,
+ GAsyncResult *res,
+ GError **error)
+{
+ NbdServerSession *self = NBD_SERVER_SESSION(initable);
+
+ return nbd_server_session_negotiate_finish(self, res, error);
+}
+
+static void
+async_initable_iface_init(GAsyncInitableIface *iface)
+{
+ iface->init_async = init_async;
+ iface->init_finish = init_finish;
+}
diff --git a/gtk/nbd/nbd-server-session.h b/gtk/nbd/nbd-server-session.h
new file mode 100644
index 0000000..bcd316f
--- /dev/null
+++ b/gtk/nbd/nbd-server-session.h
@@ -0,0 +1,56 @@
+/* -*- Mode: C; c-basic-offset: 4; indent-tabs-mode: nil -*- */
+/*
+ Copyright (C) 2013 Red Hat, Inc.
+
+ 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.1 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, see <http://www.gnu.org/licenses/>.
+*/
+
+#ifndef _NBD_SERVER_SESSION_H_
+# define _NBD_SERVER_SESSION_H_
+
+#include <gio/gio.h>
+#include "nbd-server.h"
+
+G_BEGIN_DECLS
+
+#define NBD_TYPE_SERVER_SESSION (nbd_server_session_get_type ())
+#define NBD_SERVER_SESSION(session) (G_TYPE_CHECK_INSTANCE_CAST ((session), NBD_TYPE_SERVER_SESSION, NbdServerSession))
+#define NBD_SERVER_SESSION_CLASS(klass) (G_TYPE_CHECK_CLASS_CAST ((klass), NBD_TYPE_SERVER_SESSION, NbdServerSessionClass))
+#define NBD_IS_SERVER_SESSION(session) (G_TYPE_CHECK_INSTANCE_TYPE ((session), NBD_TYPE_SERVER_SESSION))
+#define NBD_IS_SERVER_SESSION_CLASS(klass) (G_TYPE_CHECK_CLASS_TYPE ((klass), NBD_TYPE_SERVER_SESSION))
+#define NBD_SERVER_SESSION_GET_CLASS(session) (G_TYPE_INSTANCE_GET_CLASS ((session), NBD_TYPE_SERVER_SESSION, NbdServerSessionClass))
+
+typedef struct _NbdServerSessionClass NbdServerSessionClass;
+typedef struct _NbdServerSession NbdServerSession;
+
+GType nbd_server_session_get_type (void) G_GNUC_CONST;
+void nbd_server_session_new (NbdServer *server,
+ GIOStream *stream,
+ NbdExport *export,
+ GCancellable *cancellable,
+ GAsyncReadyCallback callback,
+ gpointer user_data);
+NbdServerSession* nbd_server_session_new_finish (GAsyncResult *result,
+ GError **error);
+void nbd_server_session_close_async (NbdServerSession *session,
+ GCancellable *cancellable,
+ GAsyncReadyCallback callback,
+ gpointer user_data);
+gboolean nbd_server_session_close_finish (NbdServerSession *session,
+ GAsyncResult *result,
+ GError **error);
+
+G_END_DECLS
+
+#endif /* _NBD_SERVER_SESSION_H_ */
diff --git a/gtk/nbd/nbd-server.c b/gtk/nbd/nbd-server.c
new file mode 100644
index 0000000..45ab274
--- /dev/null
+++ b/gtk/nbd/nbd-server.c
@@ -0,0 +1,108 @@
+/* -*- Mode: C; c-basic-offset: 4; indent-tabs-mode: nil -*- */
+/*
+ Copyright (C) 2013 Red Hat, Inc.
+
+ 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.1 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, see <http://www.gnu.org/licenses/>.
+*/
+
+#include "nbd-server.h"
+#include "nbd-priv.h"
+
+struct _NbdServerClass
+{
+ GObjectClass parent_class;
+};
+
+struct _NbdServer
+{
+ GObject parent_instance;
+
+ GHashTable *exports;
+};
+
+enum {
+ PROP_0,
+};
+
+G_DEFINE_TYPE(NbdServer, nbd_server, G_TYPE_OBJECT);
+
+static void
+nbd_server_init(NbdServer *self)
+{
+ self->exports = g_hash_table_new_full(g_str_hash,
+ g_str_equal,
+ NULL,
+ g_object_unref);
+}
+
+static void
+nbd_server_dispose(GObject *object)
+{
+ NbdServer *self = NBD_SERVER(object);
+
+ g_clear_pointer(&self->exports, g_hash_table_unref);
+
+ G_OBJECT_CLASS(nbd_server_parent_class)->dispose(object);
+}
+
+static void
+nbd_server_finalize(GObject *object)
+{
+ NbdServer *self = NBD_SERVER(object);
+
+ G_OBJECT_CLASS(nbd_server_parent_class)->finalize(object);
+}
+
+static void
+nbd_server_class_init(NbdServerClass *klass)
+{
+ GObjectClass* object_class = G_OBJECT_CLASS (klass);
+
+ object_class->finalize = nbd_server_finalize;
+ object_class->dispose = nbd_server_dispose;
+}
+
+NbdServer *
+nbd_server_new(void)
+{
+ return g_object_new(NBD_TYPE_SERVER, NULL);
+}
+
+void
+nbd_server_add_export(NbdServer *self, NbdExport *export)
+{
+ g_return_if_fail(NBD_IS_SERVER(self));
+ g_return_if_fail(NBD_IS_EXPORT(export));
+
+ g_hash_table_replace(self->exports,
+ (gpointer)nbd_export_get_name(export), g_object_ref(export));
+}
+
+gboolean
+nbd_server_remove_export(NbdServer *self, NbdExport *export)
+{
+ g_return_val_if_fail(NBD_IS_SERVER(self), FALSE);
+ g_return_val_if_fail(NBD_IS_EXPORT(export), FALSE);
+
+ return g_hash_table_remove(self->exports, nbd_export_get_name(export));
+}
+
+NbdExport *
+nbd_server_get_export(NbdServer *self, gchar *name)
+{
+ g_return_val_if_fail(NBD_IS_SERVER(self), NULL);
+ g_return_val_if_fail(name != NULL, NULL);
+
+ return g_hash_table_lookup(self->exports, name);
+}
diff --git a/gtk/nbd/nbd-server.h b/gtk/nbd/nbd-server.h
new file mode 100644
index 0000000..026347a
--- /dev/null
+++ b/gtk/nbd/nbd-server.h
@@ -0,0 +1,45 @@
+/* -*- Mode: C; c-basic-offset: 4; indent-tabs-mode: nil -*- */
+/*
+ Copyright (C) 2013 Red Hat, Inc.
+
+ 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.1 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, see <http://www.gnu.org/licenses/>.
+*/
+
+#ifndef _NBD_SERVER_H_
+# define _NBD_SERVER_H_
+
+#include <gio/gio.h>
+#include "nbd-export.h"
+
+G_BEGIN_DECLS
+
+#define NBD_TYPE_SERVER (nbd_server_get_type ())
+#define NBD_SERVER(server) (G_TYPE_CHECK_INSTANCE_CAST ((server), NBD_TYPE_SERVER, NbdServer))
+#define NBD_SERVER_CLASS(klass) (G_TYPE_CHECK_CLASS_CAST ((klass), NBD_TYPE_SERVER, NbdServerClass))
+#define NBD_IS_SERVER(server) (G_TYPE_CHECK_INSTANCE_TYPE ((server), NBD_TYPE_SERVER))
+#define NBD_IS_SERVER_CLASS(klass) (G_TYPE_CHECK_CLASS_TYPE ((klass), NBD_TYPE_SERVER))
+#define NBD_SERVER_GET_CLASS(server) (G_TYPE_INSTANCE_GET_CLASS ((server), NBD_TYPE_SERVER, NbdServerClass))
+
+typedef struct _NbdServerClass NbdServerClass;
+typedef struct _NbdServer NbdServer;
+
+GType nbd_server_get_type (void) G_GNUC_CONST;
+NbdServer* nbd_server_new (void);
+void nbd_server_add_export (NbdServer *server, NbdExport *export);
+gboolean nbd_server_remove_export (NbdServer *server, NbdExport *export);
+NbdExport* nbd_server_get_export (NbdServer *server, gchar *name);
+
+G_END_DECLS
+
+#endif /* _NBD_SERVER_H_ */
diff --git a/gtk/nbd/nbd.c b/gtk/nbd/nbd.c
new file mode 100644
index 0000000..866da2f
--- /dev/null
+++ b/gtk/nbd/nbd.c
@@ -0,0 +1,25 @@
+/* -*- Mode: C; c-basic-offset: 4; indent-tabs-mode: nil -*- */
+/*
+ Copyright (C) 2013 Red Hat, Inc.
+
+ 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.1 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, see <http://www.gnu.org/licenses/>.
+*/
+
+#include "nbd.h"
+
+GQuark
+nbd_error_quark(void)
+{
+ return g_quark_from_static_string("nbd-error-quark");
+}
diff --git a/gtk/nbd/nbd.h b/gtk/nbd/nbd.h
new file mode 100644
index 0000000..4b173ff
--- /dev/null
+++ b/gtk/nbd/nbd.h
@@ -0,0 +1,47 @@
+/* -*- Mode: C; c-basic-offset: 4; indent-tabs-mode: nil -*- */
+/*
+ Copyright (C) 2013 Red Hat, Inc.
+
+ 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.1 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, see <http://www.gnu.org/licenses/>.
+*/
+
+#ifndef _NBD_H_
+# define _NBD_H_
+
+#include <gio/gio.h>
+
+#include "nbd-enums.h"
+#include "nbd-export.h"
+#include "nbd-server-session.h"
+#include "nbd-server.h"
+
+G_BEGIN_DECLS
+
+#define NBD_ERROR nbd_error_quark ()
+GQuark nbd_error_quark(void);
+
+/**
+ * NbdError:
+ * @NBD_ERROR_FAILED: generic error code
+ *
+ * Error codes returned by NBD API.
+ */
+enum
+{
+ NBD_ERROR_FAILED = 1,
+};
+
+G_END_DECLS
+
+#endif /* _NBD_H_ */