This channel implements the NBD protocol and the port events defined by Spice protocol. --- doc/reference/spice-gtk-docs.xml | 1 + doc/reference/spice-gtk-sections.txt | 18 ++ doc/reference/spice-gtk.types | 3 +- gtk/Makefile.am | 16 +- gtk/channel-nbd.c | 425 +++++++++++++++++++++++++++++++++++ gtk/channel-nbd.h | 88 ++++++++ gtk/map-file | 5 + gtk/spice-channel.c | 6 + gtk/spice-client.h | 1 + gtk/spice-glib-sym-file | 4 + 10 files changed, 564 insertions(+), 3 deletions(-) create mode 100644 gtk/channel-nbd.c create mode 100644 gtk/channel-nbd.h diff --git a/doc/reference/spice-gtk-docs.xml b/doc/reference/spice-gtk-docs.xml index 4a9a3cf..b7cea5a 100644 --- a/doc/reference/spice-gtk-docs.xml +++ b/doc/reference/spice-gtk-docs.xml @@ -37,6 +37,7 @@ <xi:include href="xml/channel-smartcard.xml"/> <xi:include href="xml/channel-usbredir.xml"/> <xi:include href="xml/channel-port.xml"/> + <xi:include href="xml/channel-nbd.xml"/> </chapter> <chapter> diff --git a/doc/reference/spice-gtk-sections.txt b/doc/reference/spice-gtk-sections.txt index b15e4bb..111f51f 100644 --- a/doc/reference/spice-gtk-sections.txt +++ b/doc/reference/spice-gtk-sections.txt @@ -425,3 +425,21 @@ SPICE_PORT_CHANNEL_GET_CLASS SpicePortChannelPrivate </SECTION> +<SECTION> +<FILE>channel-nbd</FILE> +<TITLE>SpiceNbdChannel</TITLE> +SpiceNbdChannel +SpiceNbdChannelClass +<SUBSECTION> +<SUBSECTION Standard> +SPICE_NBD_CHANNEL +SPICE_IS_NBD_CHANNEL +SPICE_TYPE_NBD_CHANNEL +spice_nbd_channel_get_type +SPICE_NBD_CHANNEL_CLASS +SPICE_IS_NBD_CHANNEL_CLASS +SPICE_NBD_CHANNEL_GET_CLASS +<SUBSECTION Private> +SpiceNbdChannelPrivate +</SECTION> + diff --git a/doc/reference/spice-gtk.types b/doc/reference/spice-gtk.types index 2f52845..29741fe 100644 --- a/doc/reference/spice-gtk.types +++ b/doc/reference/spice-gtk.types @@ -42,4 +42,5 @@ spice_usbredir_channel_get_type spice_usb_device_get_type spice_usb_device_manager_get_type spice_usb_device_widget_get_type -spice_port_channel_get_type \ No newline at end of file +spice_port_channel_get_type +spice_nbd_channel_get_type \ No newline at end of file diff --git a/gtk/Makefile.am b/gtk/Makefile.am index 1932b46..7998262 100644 --- a/gtk/Makefile.am +++ b/gtk/Makefile.am @@ -76,6 +76,7 @@ SPICE_COMMON_CPPFLAGS = \ -DUSB_IDS=\""$(USB_IDS)"\" \ -DSPICE_DISABLE_ABORT \ -I$(top_srcdir) \ + -I$(srcdir)/nbd \ $(COMMON_CFLAGS) \ $(PIXMAN_CFLAGS) \ $(CELT051_CFLAGS) \ @@ -179,6 +180,7 @@ libspice_client_glib_2_0_la_LDFLAGS = \ libspice_client_glib_2_0_la_LIBADD = \ $(top_builddir)/spice-common/common/libspice-common.la \ $(top_builddir)/spice-common/common/libspice-common-client.la \ + nbd/libnbd.la \ $(GLIB2_LIBS) \ $(GIO_LIBS) \ $(GOBJECT2_LIBS) \ @@ -229,6 +231,7 @@ libspice_client_glib_2_0_la_SOURCES = \ gio-coroutine.h \ \ channel-base.c \ + channel-nbd.c \ channel-cursor.c \ channel-display.c \ channel-display-priv.h \ @@ -447,11 +450,19 @@ spice-marshal.c: spice-marshal.txt spice-marshal.h: spice-marshal.txt $(AM_V_GEN)glib-genmarshal --header $< > $@ || (rm -f $@ && exit 1) -spice-glib-enums.c: spice-channel.h channel-inputs.h spice-session.h +SPICE_GLIB_ENUMS_FILES = \ + channel-nbd.h \ + channel-inputs.h \ + spice-channel.h \ + spice-session.h \ + $(NULL) + +spice-glib-enums.c: $(SPICE_GLIB_ENUMS_FILES) $(AM_V_GEN)glib-mkenums --fhead "#include <glib-object.h>\n" \ --fhead "#include \"spice-glib-enums.h\"\n\n" \ --fprod "\n#include \"spice-session.h\"\n" \ --fprod "\n#include \"spice-channel.h\"\n" \ + --fprod "\n#include \"channel-nbd.h\"\n" \ --fprod "\n#include \"channel-inputs.h\"\n" \ --vhead "static const G@Type@Value _@enum_name@_values[] = {" \ --vprod " { @VALUENAME@, \"@VALUENAME@\", \"@valuenick@\" }," \ @@ -466,7 +477,7 @@ spice-glib-enums.c: spice-channel.h channel-inputs.h spice-session.h --vtail " return type;\n}\n\n" \ $^ > $@ -spice-glib-enums.h: spice-channel.h channel-inputs.h spice-session.h +spice-glib-enums.h: $(SPICE_GLIB_ENUMS_FILES) $(AM_V_GEN)glib-mkenums --fhead "#ifndef SPICE_GLIB_ENUMS_H\n" \ --fhead "#define SPICE_GLIB_ENUMS_H\n\n" \ --fhead "G_BEGIN_DECLS\n\n" \ @@ -589,6 +600,7 @@ glib_introspection_files = \ spice-glib-enums.c \ spice-option.c \ spice-util.c \ + channel-nbd.c \ channel-cursor.c \ channel-display.c \ channel-inputs.c \ diff --git a/gtk/channel-nbd.c b/gtk/channel-nbd.c new file mode 100644 index 0000000..e5978fd --- /dev/null +++ b/gtk/channel-nbd.c @@ -0,0 +1,425 @@ +/* -*- 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 "spice-client.h" +#include "spice-common.h" +#include "spice-channel-priv.h" +#include "spice-marshal.h" +#include "glib-compat.h" +#include "nbd/nbd.h" +#include "vmcstream.h" + +/** + * SECTION:channel-nbd + * @short_description: blockdevice communication channel + * @title: Block device Channel + * @section_id: + * @see_also: #SpiceChannel + * @stability: Stable + * @include: channel-nbd.h + * + * The "nbd" channel exports block device to a Spice server, using the + * Network Block Device protocol. This is mainly useful to redirect + * ISO images to a CD device for example. + * + * As of today, the implementation allows read-only operations. + * + * Since: 0.20 + */ + +#define SPICE_NBD_CHANNEL_GET_PRIVATE(obj) \ + (G_TYPE_INSTANCE_GET_PRIVATE((obj), SPICE_TYPE_NBD_CHANNEL, SpiceNbdChannelPrivate)) + +struct _SpiceNbdChannelPrivate { + NbdServer *server; + + gboolean pending_update; + gboolean pending_session; + NbdServerSession *session; + + GCancellable *cancellable; + NbdExport *export; + SpiceVmcStream *stream; +}; + +G_DEFINE_TYPE(SpiceNbdChannel, spice_nbd_channel, SPICE_TYPE_PORT_CHANNEL) + +/* Properties */ +enum { + PROP_0, + + PROP_NBD_FILE, + PROP_NBD_OPENED, +}; + +static void spice_nbd_handle_msg(SpiceChannel *channel, SpiceMsgIn *msg); +static void update_nbd_session(SpiceNbdChannel *self); + +static void +session_closed(GObject *source_object, + GAsyncResult *res, + gpointer user_data) +{ + SpiceNbdChannel *self = user_data; + SpiceNbdChannelPrivate *c = self->priv; + GError *error = NULL; + + CHANNEL_DEBUG(self, "session closed"); + + if (!nbd_server_session_close_finish(c->session, res, &error)) { + g_warning("%s", error->message); + g_clear_error(&error); + } + + g_clear_object(&c->session); + g_object_notify(G_OBJECT(self), "nbd-opened"); + + c->pending_update = FALSE; + update_nbd_session(self); +} + +static void port_event(SpiceNbdChannel *self, gint event) +{ + SpiceNbdChannelPrivate *c = self->priv; + + CHANNEL_DEBUG(self, "port event:%d pending:%d", event, c->pending_update); + if (event != SPICE_PORT_EVENT_OPENED || + !c->pending_update) + return; + + if (c->session) + nbd_server_session_close_async(c->session, + NULL, + session_closed, + self); + else { + c->pending_update = FALSE; + update_nbd_session(self); + } + +} + +static void spice_nbd_channel_init(SpiceNbdChannel *channel) +{ + SpiceNbdChannelPrivate *c = SPICE_NBD_CHANNEL_GET_PRIVATE(channel); + + channel->priv = c; + + c->stream = spice_vmc_stream_new(SPICE_CHANNEL(channel)); + c->server = nbd_server_new(); + c->cancellable = g_cancellable_new(); +} + +/** + * spice_nbd_channel_get_file: + * @channel: a %SpiceNbdChannel + * + * Returns the %GFile that has been set with + * spice_nbd_channel_set_file_async(). + * + * Returns: (transfer none): The associated %GFile or %NULL. + **/ +GFile* spice_nbd_channel_get_file(SpiceNbdChannel *self) +{ + GFile *file = NULL; + + g_return_val_if_fail(SPICE_IS_NBD_CHANNEL(self), NULL); + SpiceNbdChannelPrivate *c = self->priv; + + if (c->export) + file = nbd_export_get_file(c->export); + + return file; +} + +static void spice_nbd_get_property(GObject *object, + guint prop_id, + GValue *value, + GParamSpec *pspec) +{ + g_return_if_fail(SPICE_IS_NBD_CHANNEL(object)); + SpiceNbdChannel *self = SPICE_NBD_CHANNEL(object); + SpiceNbdChannelPrivate *c = self->priv; + + switch (prop_id) { + case PROP_NBD_FILE: + g_value_set_object(value, spice_nbd_channel_get_file(self)); + break; + case PROP_NBD_OPENED: + g_value_set_boolean(value, !!c->session); + break; + default: + G_OBJECT_WARN_INVALID_PROPERTY_ID(object, prop_id, pspec); + break; + } +} + +static void spice_nbd_channel_finalize(GObject *object) +{ + SpiceNbdChannelPrivate *c = SPICE_NBD_CHANNEL(object)->priv; + + g_clear_object(&c->export); + g_clear_object(&c->server); + g_clear_object(&c->session); + g_clear_object(&c->cancellable); + + G_OBJECT_CLASS(spice_nbd_channel_parent_class)->finalize(object); +} + +static void +session_ready(GObject *source_object, + GAsyncResult *res, + gpointer user_data) +{ + SpiceNbdChannel *self = user_data; + SpiceNbdChannelPrivate *c = self->priv; + GError *error = NULL; + + g_warn_if_fail(c->session == NULL); + c->session = nbd_server_session_new_finish(res, &error); + CHANNEL_DEBUG(self, "New NBD session %p", c->session); + c->pending_session = FALSE; + + if (!c->session) { + if (error) + g_warning("%s", error->message); + g_clear_error(&error); + } else + g_object_notify(G_OBJECT(self), "nbd-opened"); +} + +static void update_nbd_session(SpiceNbdChannel *self) +{ + SpiceNbdChannelPrivate *c = self->priv; + + if (spice_channel_get_state(SPICE_CHANNEL(self)) != SPICE_CHANNEL_STATE_READY || + !c->export || + c->pending_update || + c->pending_session) + return; + + if (c->session) { + g_message("PORTEVENT"); + spice_port_event(SPICE_PORT_CHANNEL(self), SPICE_PORT_EVENT_BREAK); + c->pending_update = TRUE; + return; + } + + c->pending_session = TRUE; + nbd_server_session_new(c->server, + G_IO_STREAM(c->stream), c->export, + c->cancellable, session_ready, self); +} + +static void spice_nbd_channel_up(SpiceChannel *channel) +{ + SpiceNbdChannel *self = SPICE_NBD_CHANNEL(channel); + + update_nbd_session(self); +} + +static void spice_nbd_channel_class_init(SpiceNbdChannelClass *klass) +{ + GObjectClass *gobject_class = G_OBJECT_CLASS(klass); + SpiceChannelClass *channel_class = SPICE_CHANNEL_CLASS(klass); + + gobject_class->finalize = spice_nbd_channel_finalize; + gobject_class->get_property = spice_nbd_get_property; + channel_class->handle_msg = spice_nbd_handle_msg; + channel_class->channel_up = spice_nbd_channel_up; + + g_signal_override_class_handler("port-event", + SPICE_TYPE_NBD_CHANNEL, + G_CALLBACK(port_event)); + + g_object_class_install_property + (gobject_class, PROP_NBD_FILE, + g_param_spec_string("nbd-file", + "file", + "Associated block device file", + NULL, + G_PARAM_READABLE | G_PARAM_STATIC_STRINGS)); + + g_object_class_install_property + (gobject_class, PROP_NBD_OPENED, + g_param_spec_boolean("nbd-opened", + "opened", + "Session opened", + FALSE, + G_PARAM_READABLE | G_PARAM_STATIC_STRINGS)); + + g_type_class_add_private(klass, sizeof(SpiceNbdChannelPrivate)); +} + +static void +export_ready(GObject *source_object, + GAsyncResult *res, + gpointer user_data) +{ + GError *error = NULL; + GSimpleAsyncResult *simple = user_data; + SpiceNbdChannel *self = SPICE_NBD_CHANNEL(g_async_result_get_source_object(G_ASYNC_RESULT(simple))); + SpiceNbdChannelPrivate *c = self->priv; + + NbdExport *export = nbd_export_new_finish(res, &error); + CHANNEL_DEBUG(self, "set export %p", export); + if (!export) { + g_simple_async_result_take_error(simple, error); + goto end; + } + + if (c->export) + nbd_server_remove_export(c->server, c->export); + + nbd_server_add_export(c->server, export); + + g_clear_object(&c->export); + c->export = export; + g_object_notify(G_OBJECT(self), "nbd-file"); + + update_nbd_session(self); + g_simple_async_result_set_op_res_gboolean(simple, TRUE); + +end: + g_simple_async_result_complete(simple); + g_object_unref(simple); + g_object_unref(self); +} + +/** + * spice_nbd_set_file_async: + * @channel: a #SpiceNbdChannel + * @file: the block device image file + * @flags: open options + * @cancellable: (allow-none): optional GCancellable object, NULL to ignore + * @callback: (scope async): callback to call when the request is satisfied + * @user_data: (closure): the data to pass to callback function + * + * Update the @file image associated with @channel. + * + * The caller is responsible to finish any previous call, no + * concurrent call will be permitted. + **/ +void +spice_nbd_channel_set_file_async(SpiceNbdChannel *self, + GFile *file, SpiceNbdOpenFlags flags, + GCancellable *cancellable, + GAsyncReadyCallback callback, + gpointer user_data) +{ + GSimpleAsyncResult *simple; + + g_return_if_fail(SPICE_IS_NBD_CHANNEL(self)); + g_return_if_fail(!file || G_IS_FILE(file)); + + SpiceNbdChannelPrivate *c = self->priv; + + if (c->pending_session) { + g_simple_async_report_error_in_idle(G_OBJECT (self), + callback, + user_data, + SPICE_CLIENT_ERROR, + SPICE_CLIENT_ERROR_FAILED, + "Channel has already a pending session"); + return; + } + + simple = g_simple_async_result_new(G_OBJECT(self), + callback, + user_data, + spice_nbd_channel_set_file_async); + + CHANNEL_DEBUG(self, "set file %p 0x%X", file, flags); + + if (!file) { + g_clear_object(&c->export); + g_object_notify(G_OBJECT(self), "nbd-file"); + + update_nbd_session(self); + + g_simple_async_result_set_op_res_gboolean(simple, TRUE); + g_simple_async_result_complete_in_idle(simple); + g_object_unref(simple); + } else + nbd_export_new("", file, flags, cancellable, export_ready, simple); +} + +/** + * spice_nbd_set_file_finish: + * @channel: a #SpiceNbdChannel + * @result: a #GAsyncResult + * @error: a #GError location to store the error occurring, or %NULL + * to ignore + * + * Finishes an asynchronous file read operation started with + * spice_nbd_set_file_async(). + * + * Returns: %TRUE on success. + **/ +gboolean +spice_nbd_channel_set_file_finish(SpiceNbdChannel *self, + GAsyncResult *result, + GError **error) +{ + GSimpleAsyncResult *simple; + + g_return_val_if_fail(SPICE_IS_NBD_CHANNEL(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), + spice_nbd_channel_set_file_async), + 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); +} + +/* coroutine context */ +static void nbd_handle_msg(SpiceChannel *channel, SpiceMsgIn *in) +{ + SpiceNbdChannel *self = SPICE_NBD_CHANNEL(channel); + SpiceNbdChannelPrivate *c = self->priv; + int size; + uint8_t *buf; + + buf = spice_msg_in_raw(in, &size); + + spice_vmc_input_stream_co_data( + SPICE_VMC_INPUT_STREAM(g_io_stream_get_input_stream(G_IO_STREAM(c->stream))), + buf, size); +} + + +/* coroutine context */ +static void spice_nbd_handle_msg(SpiceChannel *channel, SpiceMsgIn *msg) +{ + int type = spice_msg_in_type(msg); + SpiceChannelClass *parent_class; + + parent_class = SPICE_CHANNEL_CLASS(spice_nbd_channel_parent_class); + + if (type == SPICE_MSG_SPICEVMC_DATA) + nbd_handle_msg(channel, msg); + else if (parent_class->handle_msg) + parent_class->handle_msg(channel, msg); + else + g_return_if_reached(); +} diff --git a/gtk/channel-nbd.h b/gtk/channel-nbd.h new file mode 100644 index 0000000..2cb2d7a --- /dev/null +++ b/gtk/channel-nbd.h @@ -0,0 +1,88 @@ +/* -*- 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 __SPICE_CLIENT_NBD_CHANNEL_H__ +#define __SPICE_CLIENT_NBD_CHANNEL_H__ + +#include <gio/gio.h> +#include "spice-client.h" +#include "channel-port.h" + +G_BEGIN_DECLS + +#define SPICE_TYPE_NBD_CHANNEL (spice_nbd_channel_get_type()) +#define SPICE_NBD_CHANNEL(obj) (G_TYPE_CHECK_INSTANCE_CAST((obj), SPICE_TYPE_NBD_CHANNEL, SpiceNbdChannel)) +#define SPICE_NBD_CHANNEL_CLASS(klass) (G_TYPE_CHECK_CLASS_CAST((klass), SPICE_TYPE_NBD_CHANNEL, SpiceNbdChannelClass)) +#define SPICE_IS_NBD_CHANNEL(obj) (G_TYPE_CHECK_INSTANCE_TYPE((obj), SPICE_TYPE_NBD_CHANNEL)) +#define SPICE_IS_NBD_CHANNEL_CLASS(klass) (G_TYPE_CHECK_CLASS_TYPE((klass), SPICE_TYPE_NBD_CHANNEL)) +#define SPICE_NBD_CHANNEL_GET_CLASS(obj) (G_TYPE_INSTANCE_GET_CLASS((obj), SPICE_TYPE_NBD_CHANNEL, SpiceNbdChannelClass)) + +typedef struct _SpiceNbdChannel SpiceNbdChannel; +typedef struct _SpiceNbdChannelClass SpiceNbdChannelClass; +typedef struct _SpiceNbdChannelPrivate SpiceNbdChannelPrivate; + +/** + * SpiceNbdOpenFlags: + * + **/ +typedef enum +{ + SPICE_NBD_OPEN_NONE = 0, + + SPICE_NBD_OPEN_READWRITE = 1 << 0, +} SpiceNbdOpenFlags; + +/** + * SpiceNbdChannel: + * + * The #SpiceNbdChannel struct is opaque and should not be accessed directly. + */ +struct _SpiceNbdChannel { + SpicePortChannel parent; + + /*< private >*/ + SpiceNbdChannelPrivate *priv; + /* Do not add fields to this struct */ +}; + +/** + * SpiceNbdChannelClass: + * @parent_class: Parent class. + * + * Class structure for #SpiceNbdChannel. + */ +struct _SpiceNbdChannelClass { + SpicePortChannelClass parent_class; + + /*< private >*/ + /* Do not add fields to this struct */ +}; + +GType spice_nbd_channel_get_type(void); + +void spice_nbd_channel_set_file_async(SpiceNbdChannel *channel, + GFile *file, SpiceNbdOpenFlags flags, + GCancellable *cancellable, + GAsyncReadyCallback callback, + gpointer user_data); +gboolean spice_nbd_channel_set_file_finish(SpiceNbdChannel *channel, + GAsyncResult *result, GError **error); +GFile* spice_nbd_channel_get_file(SpiceNbdChannel *channel); + +G_END_DECLS + +#endif /* __SPICE_CLIENT_NBD_CHANNEL_H__ */ diff --git a/gtk/map-file b/gtk/map-file index 558a4d7..053372a 100644 --- a/gtk/map-file +++ b/gtk/map-file @@ -3,6 +3,11 @@ global: spice_audio_get; spice_audio_get_type; spice_audio_new; +spice_nbd_channel_get_type; +spice_nbd_open_flags_get_type; +spice_nbd_channel_set_file_async; +spice_nbd_channel_set_file_finish; +spice_nbd_channel_get_file; spice_channel_connect; spice_channel_destroy; spice_channel_disconnect; diff --git a/gtk/spice-channel.c b/gtk/spice-channel.c index 0a32d6c..5630741 100644 --- a/gtk/spice-channel.c +++ b/gtk/spice-channel.c @@ -1882,6 +1882,7 @@ static const char *to_string[] = { [ SPICE_CHANNEL_SMARTCARD ] = "smartcard", [ SPICE_CHANNEL_USBREDIR ] = "usbredir", [ SPICE_CHANNEL_PORT ] = "port", + [ SPICE_CHANNEL_NBD ] = "nbd", }; const gchar* spice_channel_type_to_string(gint type) @@ -1924,6 +1925,7 @@ gchar *spice_channel_supported_string(void) #ifdef USE_USBREDIR spice_channel_type_to_string(SPICE_CHANNEL_USBREDIR), #endif + spice_channel_type_to_string(SPICE_CHANNEL_NBD), NULL); } @@ -1988,6 +1990,10 @@ SpiceChannel *spice_channel_new(SpiceSession *s, int type, int id) break; } #endif + case SPICE_CHANNEL_NBD: { + gtype = SPICE_TYPE_NBD_CHANNEL; + break; + } case SPICE_CHANNEL_PORT: gtype = SPICE_TYPE_PORT_CHANNEL; break; diff --git a/gtk/spice-client.h b/gtk/spice-client.h index 730d11a..5b9b01b 100644 --- a/gtk/spice-client.h +++ b/gtk/spice-client.h @@ -41,6 +41,7 @@ #include "channel-smartcard.h" #include "channel-usbredir.h" #include "channel-port.h" +#include "channel-nbd.h" #include "smartcard-manager.h" #include "usb-device-manager.h" diff --git a/gtk/spice-glib-sym-file b/gtk/spice-glib-sym-file index 5a580d6..269a675 100644 --- a/gtk/spice-glib-sym-file +++ b/gtk/spice-glib-sym-file @@ -1,6 +1,10 @@ spice_audio_get spice_audio_get_type spice_audio_new +spice_nbd_channel_get_type +spice_nbd_open_flags_get_type +spice_nbd_set_file_async +spice_nbd_set_file_finish spice_channel_connect spice_channel_destroy spice_channel_disconnect -- 1.8.3.rc1.49.g8d97506 _______________________________________________ Spice-devel mailing list Spice-devel@xxxxxxxxxxxxxxxxxxxxx http://lists.freedesktop.org/mailman/listinfo/spice-devel