2008-07-03 13:41:03 +00:00
|
|
|
/*
|
|
|
|
* QEMU Block driver for NBD
|
|
|
|
*
|
|
|
|
* Copyright (C) 2008 Bull S.A.S.
|
2008-07-08 18:57:05 +00:00
|
|
|
* Author: Laurent Vivier <Laurent.Vivier@bull.net>
|
2008-07-03 13:41:03 +00:00
|
|
|
*
|
|
|
|
* Some parts:
|
|
|
|
* Copyright (C) 2007 Anthony Liguori <anthony@codemonkey.ws>
|
|
|
|
*
|
|
|
|
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
|
|
* of this software and associated documentation files (the "Software"), to deal
|
|
|
|
* in the Software without restriction, including without limitation the rights
|
|
|
|
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
|
|
* copies of the Software, and to permit persons to whom the Software is
|
|
|
|
* furnished to do so, subject to the following conditions:
|
|
|
|
*
|
|
|
|
* The above copyright notice and this permission notice shall be included in
|
|
|
|
* all copies or substantial portions of the Software.
|
|
|
|
*
|
|
|
|
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
|
|
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
|
|
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
|
|
|
|
* THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
|
|
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
|
|
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
|
|
|
|
* THE SOFTWARE.
|
|
|
|
*/
|
|
|
|
|
2016-01-18 18:01:42 +00:00
|
|
|
#include "qemu/osdep.h"
|
2013-12-01 21:23:41 +00:00
|
|
|
#include "block/nbd-client.h"
|
include/qemu/osdep.h: Don't include qapi/error.h
Commit 57cb38b included qapi/error.h into qemu/osdep.h to get the
Error typedef. Since then, we've moved to include qemu/osdep.h
everywhere. Its file comment explains: "To avoid getting into
possible circular include dependencies, this file should not include
any other QEMU headers, with the exceptions of config-host.h,
compiler.h, os-posix.h and os-win32.h, all of which are doing a
similar job to this file and are under similar constraints."
qapi/error.h doesn't do a similar job, and it doesn't adhere to
similar constraints: it includes qapi-types.h. That's in excess of
100KiB of crap most .c files don't actually need.
Add the typedef to qemu/typedefs.h, and include that instead of
qapi/error.h. Include qapi/error.h in .c files that need it and don't
get it now. Include qapi-types.h in qom/object.h for uint16List.
Update scripts/clean-includes accordingly. Update it further to match
reality: replace config.h by config-target.h, add sysemu/os-posix.h,
sysemu/os-win32.h. Update the list of includes in the qemu/osdep.h
comment quoted above similarly.
This reduces the number of objects depending on qapi/error.h from "all
of them" to less than a third. Unfortunately, the number depending on
qapi-types.h shrinks only a little. More work is needed for that one.
Signed-off-by: Markus Armbruster <armbru@redhat.com>
[Fix compilation without the spice devel packages. - Paolo]
Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
2016-03-14 08:01:28 +00:00
|
|
|
#include "qapi/error.h"
|
2012-12-17 17:20:00 +00:00
|
|
|
#include "qemu/uri.h"
|
2012-12-17 17:19:44 +00:00
|
|
|
#include "block/block_int.h"
|
2012-12-17 17:20:00 +00:00
|
|
|
#include "qemu/module.h"
|
2014-07-18 18:24:59 +00:00
|
|
|
#include "qapi/qmp/qdict.h"
|
2013-03-07 15:15:11 +00:00
|
|
|
#include "qapi/qmp/qjson.h"
|
|
|
|
#include "qapi/qmp/qint.h"
|
2014-07-18 18:24:59 +00:00
|
|
|
#include "qapi/qmp/qstring.h"
|
2016-03-20 17:16:19 +00:00
|
|
|
#include "qemu/cutils.h"
|
2008-07-03 13:41:03 +00:00
|
|
|
|
2010-08-25 20:48:33 +00:00
|
|
|
#define EN_OPTSTR ":exportname="
|
|
|
|
|
2008-07-03 13:41:03 +00:00
|
|
|
typedef struct BDRVNBDState {
|
2013-12-01 21:23:41 +00:00
|
|
|
NbdClientSession client;
|
2008-07-03 13:41:03 +00:00
|
|
|
} BDRVNBDState;
|
|
|
|
|
2013-03-07 15:15:11 +00:00
|
|
|
static int nbd_parse_uri(const char *filename, QDict *options)
|
2012-11-04 12:04:24 +00:00
|
|
|
{
|
|
|
|
URI *uri;
|
|
|
|
const char *p;
|
|
|
|
QueryParams *qp = NULL;
|
|
|
|
int ret = 0;
|
2013-03-07 15:15:11 +00:00
|
|
|
bool is_unix;
|
2012-11-04 12:04:24 +00:00
|
|
|
|
|
|
|
uri = uri_parse(filename);
|
|
|
|
if (!uri) {
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* transport */
|
|
|
|
if (!strcmp(uri->scheme, "nbd")) {
|
2013-03-07 15:15:11 +00:00
|
|
|
is_unix = false;
|
2012-11-04 12:04:24 +00:00
|
|
|
} else if (!strcmp(uri->scheme, "nbd+tcp")) {
|
2013-03-07 15:15:11 +00:00
|
|
|
is_unix = false;
|
2012-11-04 12:04:24 +00:00
|
|
|
} else if (!strcmp(uri->scheme, "nbd+unix")) {
|
2013-03-07 15:15:11 +00:00
|
|
|
is_unix = true;
|
2012-11-04 12:04:24 +00:00
|
|
|
} else {
|
|
|
|
ret = -EINVAL;
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
|
|
|
|
p = uri->path ? uri->path : "/";
|
|
|
|
p += strspn(p, "/");
|
|
|
|
if (p[0]) {
|
2013-03-07 15:15:11 +00:00
|
|
|
qdict_put(options, "export", qstring_from_str(p));
|
2012-11-04 12:04:24 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
qp = query_params_parse(uri->query);
|
2013-03-07 15:15:11 +00:00
|
|
|
if (qp->n > 1 || (is_unix && !qp->n) || (!is_unix && qp->n)) {
|
2012-11-04 12:04:24 +00:00
|
|
|
ret = -EINVAL;
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
|
2013-03-07 15:15:11 +00:00
|
|
|
if (is_unix) {
|
2012-11-04 12:04:24 +00:00
|
|
|
/* nbd+unix:///export?socket=path */
|
|
|
|
if (uri->server || uri->port || strcmp(qp->p[0].name, "socket")) {
|
|
|
|
ret = -EINVAL;
|
|
|
|
goto out;
|
|
|
|
}
|
2013-03-07 15:15:11 +00:00
|
|
|
qdict_put(options, "path", qstring_from_str(qp->p[0].value));
|
2012-11-04 12:04:24 +00:00
|
|
|
} else {
|
2013-06-03 15:54:56 +00:00
|
|
|
QString *host;
|
2013-03-18 15:56:05 +00:00
|
|
|
/* nbd[+tcp]://host[:port]/export */
|
2012-11-04 12:04:24 +00:00
|
|
|
if (!uri->server) {
|
|
|
|
ret = -EINVAL;
|
|
|
|
goto out;
|
|
|
|
}
|
2013-03-15 10:55:29 +00:00
|
|
|
|
2013-06-03 15:54:56 +00:00
|
|
|
/* strip braces from literal IPv6 address */
|
|
|
|
if (uri->server[0] == '[') {
|
|
|
|
host = qstring_from_substr(uri->server, 1,
|
|
|
|
strlen(uri->server) - 2);
|
|
|
|
} else {
|
|
|
|
host = qstring_from_str(uri->server);
|
|
|
|
}
|
|
|
|
|
|
|
|
qdict_put(options, "host", host);
|
2013-03-18 15:56:05 +00:00
|
|
|
if (uri->port) {
|
|
|
|
char* port_str = g_strdup_printf("%d", uri->port);
|
|
|
|
qdict_put(options, "port", qstring_from_str(port_str));
|
|
|
|
g_free(port_str);
|
|
|
|
}
|
2012-11-04 12:04:24 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
out:
|
|
|
|
if (qp) {
|
|
|
|
query_params_free(qp);
|
|
|
|
}
|
|
|
|
uri_free(uri);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2013-03-15 17:47:22 +00:00
|
|
|
static void nbd_parse_filename(const char *filename, QDict *options,
|
|
|
|
Error **errp)
|
2008-07-03 13:41:03 +00:00
|
|
|
{
|
2010-08-25 20:48:33 +00:00
|
|
|
char *file;
|
2011-02-22 15:44:54 +00:00
|
|
|
char *export_name;
|
|
|
|
const char *host_spec;
|
2008-07-03 13:41:03 +00:00
|
|
|
const char *unixpath;
|
|
|
|
|
2013-03-20 18:23:23 +00:00
|
|
|
if (qdict_haskey(options, "host")
|
|
|
|
|| qdict_haskey(options, "port")
|
|
|
|
|| qdict_haskey(options, "path"))
|
|
|
|
{
|
|
|
|
error_setg(errp, "host/port/path and a file name may not be specified "
|
|
|
|
"at the same time");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2012-11-04 12:04:24 +00:00
|
|
|
if (strstr(filename, "://")) {
|
2013-03-15 17:47:22 +00:00
|
|
|
int ret = nbd_parse_uri(filename, options);
|
|
|
|
if (ret < 0) {
|
|
|
|
error_setg(errp, "No valid URL specified");
|
|
|
|
}
|
|
|
|
return;
|
2012-11-04 12:04:24 +00:00
|
|
|
}
|
|
|
|
|
2011-08-21 03:09:37 +00:00
|
|
|
file = g_strdup(filename);
|
2010-08-25 20:48:33 +00:00
|
|
|
|
2011-02-22 15:44:54 +00:00
|
|
|
export_name = strstr(file, EN_OPTSTR);
|
|
|
|
if (export_name) {
|
|
|
|
if (export_name[strlen(EN_OPTSTR)] == 0) {
|
2010-08-25 20:48:33 +00:00
|
|
|
goto out;
|
|
|
|
}
|
2011-02-22 15:44:54 +00:00
|
|
|
export_name[0] = 0; /* truncate 'file' */
|
|
|
|
export_name += strlen(EN_OPTSTR);
|
2013-03-07 15:15:11 +00:00
|
|
|
|
|
|
|
qdict_put(options, "export", qstring_from_str(export_name));
|
2010-08-25 20:48:33 +00:00
|
|
|
}
|
|
|
|
|
2011-02-22 15:44:54 +00:00
|
|
|
/* extract the host_spec - fail if it's not nbd:... */
|
|
|
|
if (!strstart(file, "nbd:", &host_spec)) {
|
2013-03-15 17:47:22 +00:00
|
|
|
error_setg(errp, "File name string for NBD must start with 'nbd:'");
|
2010-08-25 20:48:33 +00:00
|
|
|
goto out;
|
|
|
|
}
|
2008-07-03 13:41:03 +00:00
|
|
|
|
2013-03-07 15:15:11 +00:00
|
|
|
if (!*host_spec) {
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
|
2011-02-22 15:44:54 +00:00
|
|
|
/* are we a UNIX or TCP socket? */
|
|
|
|
if (strstart(host_spec, "unix:", &unixpath)) {
|
2013-03-07 15:15:11 +00:00
|
|
|
qdict_put(options, "path", qstring_from_str(unixpath));
|
2008-07-03 13:41:03 +00:00
|
|
|
} else {
|
2013-03-07 15:15:11 +00:00
|
|
|
InetSocketAddress *addr = NULL;
|
|
|
|
|
2013-03-15 17:47:22 +00:00
|
|
|
addr = inet_parse(host_spec, errp);
|
2014-04-25 14:50:33 +00:00
|
|
|
if (!addr) {
|
2013-03-15 10:55:29 +00:00
|
|
|
goto out;
|
|
|
|
}
|
2008-07-03 13:41:03 +00:00
|
|
|
|
2013-03-07 15:15:11 +00:00
|
|
|
qdict_put(options, "host", qstring_from_str(addr->host));
|
|
|
|
qdict_put(options, "port", qstring_from_str(addr->port));
|
|
|
|
qapi_free_InetSocketAddress(addr);
|
|
|
|
}
|
2008-07-03 13:41:03 +00:00
|
|
|
|
2011-02-22 15:44:54 +00:00
|
|
|
out:
|
2011-08-21 03:09:37 +00:00
|
|
|
g_free(file);
|
2013-03-07 15:15:11 +00:00
|
|
|
}
|
|
|
|
|
2016-08-15 13:29:24 +00:00
|
|
|
static SocketAddress *nbd_config(BDRVNBDState *s, QemuOpts *opts, char **export,
|
2015-09-16 13:52:22 +00:00
|
|
|
Error **errp)
|
2013-03-07 15:15:11 +00:00
|
|
|
{
|
2015-09-16 13:52:22 +00:00
|
|
|
SocketAddress *saddr;
|
2013-03-07 15:15:11 +00:00
|
|
|
|
2016-08-15 13:29:24 +00:00
|
|
|
if (!qemu_opt_get(opts, "path") == !qemu_opt_get(opts, "host")) {
|
|
|
|
if (qemu_opt_get(opts, "path")) {
|
2014-02-17 13:43:49 +00:00
|
|
|
error_setg(errp, "path and host may not be used at the same time.");
|
2014-02-17 13:43:48 +00:00
|
|
|
} else {
|
2014-02-17 13:43:49 +00:00
|
|
|
error_setg(errp, "one of path and host must be specified.");
|
2013-03-20 18:23:23 +00:00
|
|
|
}
|
2015-09-16 13:52:22 +00:00
|
|
|
return NULL;
|
2011-02-22 15:44:54 +00:00
|
|
|
}
|
2013-03-07 15:15:11 +00:00
|
|
|
|
2015-09-16 13:52:22 +00:00
|
|
|
saddr = g_new0(SocketAddress, 1);
|
2013-03-07 15:15:11 +00:00
|
|
|
|
2016-08-15 13:29:24 +00:00
|
|
|
if (qemu_opt_get(opts, "path")) {
|
2016-03-03 16:16:48 +00:00
|
|
|
UnixSocketAddress *q_unix;
|
2015-10-26 22:34:54 +00:00
|
|
|
saddr->type = SOCKET_ADDRESS_KIND_UNIX;
|
qapi: Don't special-case simple union wrappers
Simple unions were carrying a special case that hid their 'data'
QMP member from the resulting C struct, via the hack method
QAPISchemaObjectTypeVariant.simple_union_type(). But by using
the work we started by unboxing flat union and alternate
branches, coupled with the ability to visit the members of an
implicit type, we can now expose the simple union's implicit
type in qapi-types.h:
| struct q_obj_ImageInfoSpecificQCow2_wrapper {
| ImageInfoSpecificQCow2 *data;
| };
|
| struct q_obj_ImageInfoSpecificVmdk_wrapper {
| ImageInfoSpecificVmdk *data;
| };
...
| struct ImageInfoSpecific {
| ImageInfoSpecificKind type;
| union { /* union tag is @type */
| void *data;
|- ImageInfoSpecificQCow2 *qcow2;
|- ImageInfoSpecificVmdk *vmdk;
|+ q_obj_ImageInfoSpecificQCow2_wrapper qcow2;
|+ q_obj_ImageInfoSpecificVmdk_wrapper vmdk;
| } u;
| };
Doing this removes asymmetry between QAPI's QMP side and its
C side (both sides now expose 'data'), and means that the
treatment of a simple union as sugar for a flat union is now
equivalent in both languages (previously the two approaches used
a different layer of dereferencing, where the simple union could
be converted to a flat union with equivalent C layout but
different {} on the wire, or to an equivalent QMP wire form
but with different C representation). Using the implicit type
also lets us get rid of the simple_union_type() hack.
Of course, now all clients of simple unions have to adjust from
using su->u.member to using su->u.member.data; while this touches
a number of files in the tree, some earlier cleanup patches
helped minimize the change to the initialization of a temporary
variable rather than every single member access. The generated
qapi-visit.c code is also affected by the layout change:
|@@ -7393,10 +7393,10 @@ void visit_type_ImageInfoSpecific_member
| }
| switch (obj->type) {
| case IMAGE_INFO_SPECIFIC_KIND_QCOW2:
|- visit_type_ImageInfoSpecificQCow2(v, "data", &obj->u.qcow2, &err);
|+ visit_type_q_obj_ImageInfoSpecificQCow2_wrapper_members(v, &obj->u.qcow2, &err);
| break;
| case IMAGE_INFO_SPECIFIC_KIND_VMDK:
|- visit_type_ImageInfoSpecificVmdk(v, "data", &obj->u.vmdk, &err);
|+ visit_type_q_obj_ImageInfoSpecificVmdk_wrapper_members(v, &obj->u.vmdk, &err);
| break;
| default:
| abort();
Signed-off-by: Eric Blake <eblake@redhat.com>
Message-Id: <1458254921-17042-13-git-send-email-eblake@redhat.com>
Signed-off-by: Markus Armbruster <armbru@redhat.com>
2016-03-17 22:48:37 +00:00
|
|
|
q_unix = saddr->u.q_unix.data = g_new0(UnixSocketAddress, 1);
|
2016-08-15 13:29:24 +00:00
|
|
|
q_unix->path = g_strdup(qemu_opt_get(opts, "path"));
|
2015-09-16 13:52:22 +00:00
|
|
|
} else {
|
2016-03-03 16:16:48 +00:00
|
|
|
InetSocketAddress *inet;
|
2015-10-26 22:34:54 +00:00
|
|
|
saddr->type = SOCKET_ADDRESS_KIND_INET;
|
qapi: Don't special-case simple union wrappers
Simple unions were carrying a special case that hid their 'data'
QMP member from the resulting C struct, via the hack method
QAPISchemaObjectTypeVariant.simple_union_type(). But by using
the work we started by unboxing flat union and alternate
branches, coupled with the ability to visit the members of an
implicit type, we can now expose the simple union's implicit
type in qapi-types.h:
| struct q_obj_ImageInfoSpecificQCow2_wrapper {
| ImageInfoSpecificQCow2 *data;
| };
|
| struct q_obj_ImageInfoSpecificVmdk_wrapper {
| ImageInfoSpecificVmdk *data;
| };
...
| struct ImageInfoSpecific {
| ImageInfoSpecificKind type;
| union { /* union tag is @type */
| void *data;
|- ImageInfoSpecificQCow2 *qcow2;
|- ImageInfoSpecificVmdk *vmdk;
|+ q_obj_ImageInfoSpecificQCow2_wrapper qcow2;
|+ q_obj_ImageInfoSpecificVmdk_wrapper vmdk;
| } u;
| };
Doing this removes asymmetry between QAPI's QMP side and its
C side (both sides now expose 'data'), and means that the
treatment of a simple union as sugar for a flat union is now
equivalent in both languages (previously the two approaches used
a different layer of dereferencing, where the simple union could
be converted to a flat union with equivalent C layout but
different {} on the wire, or to an equivalent QMP wire form
but with different C representation). Using the implicit type
also lets us get rid of the simple_union_type() hack.
Of course, now all clients of simple unions have to adjust from
using su->u.member to using su->u.member.data; while this touches
a number of files in the tree, some earlier cleanup patches
helped minimize the change to the initialization of a temporary
variable rather than every single member access. The generated
qapi-visit.c code is also affected by the layout change:
|@@ -7393,10 +7393,10 @@ void visit_type_ImageInfoSpecific_member
| }
| switch (obj->type) {
| case IMAGE_INFO_SPECIFIC_KIND_QCOW2:
|- visit_type_ImageInfoSpecificQCow2(v, "data", &obj->u.qcow2, &err);
|+ visit_type_q_obj_ImageInfoSpecificQCow2_wrapper_members(v, &obj->u.qcow2, &err);
| break;
| case IMAGE_INFO_SPECIFIC_KIND_VMDK:
|- visit_type_ImageInfoSpecificVmdk(v, "data", &obj->u.vmdk, &err);
|+ visit_type_q_obj_ImageInfoSpecificVmdk_wrapper_members(v, &obj->u.vmdk, &err);
| break;
| default:
| abort();
Signed-off-by: Eric Blake <eblake@redhat.com>
Message-Id: <1458254921-17042-13-git-send-email-eblake@redhat.com>
Signed-off-by: Markus Armbruster <armbru@redhat.com>
2016-03-17 22:48:37 +00:00
|
|
|
inet = saddr->u.inet.data = g_new0(InetSocketAddress, 1);
|
2016-08-15 13:29:24 +00:00
|
|
|
inet->host = g_strdup(qemu_opt_get(opts, "host"));
|
|
|
|
inet->port = g_strdup(qemu_opt_get(opts, "port"));
|
|
|
|
if (!inet->port) {
|
2016-03-03 16:16:48 +00:00
|
|
|
inet->port = g_strdup_printf("%d", NBD_DEFAULT_PORT);
|
2015-09-16 13:52:22 +00:00
|
|
|
}
|
2013-03-07 15:15:11 +00:00
|
|
|
}
|
|
|
|
|
2015-10-26 22:34:54 +00:00
|
|
|
s->client.is_unix = saddr->type == SOCKET_ADDRESS_KIND_UNIX;
|
2013-03-18 15:56:05 +00:00
|
|
|
|
2016-08-15 13:29:24 +00:00
|
|
|
*export = g_strdup(qemu_opt_get(opts, "export"));
|
2015-09-16 13:52:22 +00:00
|
|
|
|
|
|
|
return saddr;
|
2011-02-22 15:44:54 +00:00
|
|
|
}
|
2010-08-25 20:48:33 +00:00
|
|
|
|
2015-02-06 21:06:16 +00:00
|
|
|
NbdClientSession *nbd_get_client_session(BlockDriverState *bs)
|
|
|
|
{
|
|
|
|
BDRVNBDState *s = bs->opaque;
|
|
|
|
return &s->client;
|
|
|
|
}
|
|
|
|
|
2016-02-10 18:41:01 +00:00
|
|
|
static QIOChannelSocket *nbd_establish_connection(SocketAddress *saddr,
|
|
|
|
Error **errp)
|
2011-02-22 15:44:54 +00:00
|
|
|
{
|
2016-02-10 18:41:01 +00:00
|
|
|
QIOChannelSocket *sioc;
|
|
|
|
Error *local_err = NULL;
|
2008-07-03 13:41:03 +00:00
|
|
|
|
2016-02-10 18:41:01 +00:00
|
|
|
sioc = qio_channel_socket_new();
|
2008-07-03 13:41:03 +00:00
|
|
|
|
2016-02-10 18:41:01 +00:00
|
|
|
qio_channel_socket_connect_sync(sioc,
|
|
|
|
saddr,
|
|
|
|
&local_err);
|
|
|
|
if (local_err) {
|
|
|
|
error_propagate(errp, local_err);
|
|
|
|
return NULL;
|
2010-08-25 20:48:33 +00:00
|
|
|
}
|
2008-07-03 13:41:03 +00:00
|
|
|
|
2016-02-10 18:41:01 +00:00
|
|
|
qio_channel_set_delay(QIO_CHANNEL(sioc), false);
|
2015-09-16 13:52:22 +00:00
|
|
|
|
2016-02-10 18:41:01 +00:00
|
|
|
return sioc;
|
2011-02-22 15:44:54 +00:00
|
|
|
}
|
|
|
|
|
2016-02-10 18:41:12 +00:00
|
|
|
|
|
|
|
static QCryptoTLSCreds *nbd_get_tls_creds(const char *id, Error **errp)
|
|
|
|
{
|
|
|
|
Object *obj;
|
|
|
|
QCryptoTLSCreds *creds;
|
|
|
|
|
|
|
|
obj = object_resolve_path_component(
|
|
|
|
object_get_objects_root(), id);
|
|
|
|
if (!obj) {
|
|
|
|
error_setg(errp, "No TLS credentials with id '%s'",
|
|
|
|
id);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
creds = (QCryptoTLSCreds *)
|
|
|
|
object_dynamic_cast(obj, TYPE_QCRYPTO_TLS_CREDS);
|
|
|
|
if (!creds) {
|
|
|
|
error_setg(errp, "Object with id '%s' is not TLS credentials",
|
|
|
|
id);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (creds->endpoint != QCRYPTO_TLS_CREDS_ENDPOINT_CLIENT) {
|
|
|
|
error_setg(errp,
|
|
|
|
"Expecting TLS credentials with a client endpoint");
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
object_ref(obj);
|
|
|
|
return creds;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2016-08-15 13:29:24 +00:00
|
|
|
static QemuOptsList nbd_runtime_opts = {
|
|
|
|
.name = "nbd",
|
|
|
|
.head = QTAILQ_HEAD_INITIALIZER(nbd_runtime_opts.head),
|
|
|
|
.desc = {
|
|
|
|
{
|
|
|
|
.name = "host",
|
|
|
|
.type = QEMU_OPT_STRING,
|
|
|
|
.help = "TCP host to connect to",
|
|
|
|
},
|
|
|
|
{
|
|
|
|
.name = "port",
|
|
|
|
.type = QEMU_OPT_STRING,
|
|
|
|
.help = "TCP port to connect to",
|
|
|
|
},
|
|
|
|
{
|
|
|
|
.name = "path",
|
|
|
|
.type = QEMU_OPT_STRING,
|
|
|
|
.help = "Unix socket path to connect to",
|
|
|
|
},
|
|
|
|
{
|
|
|
|
.name = "export",
|
|
|
|
.type = QEMU_OPT_STRING,
|
|
|
|
.help = "Name of the NBD export to open",
|
|
|
|
},
|
|
|
|
{
|
|
|
|
.name = "tls-creds",
|
|
|
|
.type = QEMU_OPT_STRING,
|
|
|
|
.help = "ID of the TLS credentials to use",
|
|
|
|
},
|
|
|
|
},
|
|
|
|
};
|
|
|
|
|
2013-09-05 12:22:29 +00:00
|
|
|
static int nbd_open(BlockDriverState *bs, QDict *options, int flags,
|
|
|
|
Error **errp)
|
2011-02-22 15:44:54 +00:00
|
|
|
{
|
|
|
|
BDRVNBDState *s = bs->opaque;
|
2016-08-15 13:29:24 +00:00
|
|
|
QemuOpts *opts = NULL;
|
|
|
|
Error *local_err = NULL;
|
2013-12-01 21:23:43 +00:00
|
|
|
char *export = NULL;
|
2016-02-10 18:41:12 +00:00
|
|
|
QIOChannelSocket *sioc = NULL;
|
2016-08-15 13:29:24 +00:00
|
|
|
SocketAddress *saddr = NULL;
|
2016-02-10 18:41:12 +00:00
|
|
|
const char *tlscredsid;
|
|
|
|
QCryptoTLSCreds *tlscreds = NULL;
|
|
|
|
const char *hostname = NULL;
|
|
|
|
int ret = -EINVAL;
|
2011-09-08 12:28:59 +00:00
|
|
|
|
2016-08-15 13:29:24 +00:00
|
|
|
opts = qemu_opts_create(&nbd_runtime_opts, NULL, 0, &error_abort);
|
|
|
|
qemu_opts_absorb_qdict(opts, options, &local_err);
|
|
|
|
if (local_err) {
|
|
|
|
error_propagate(errp, local_err);
|
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
|
2011-02-22 15:44:54 +00:00
|
|
|
/* Pop the config into our state object. Exit if invalid. */
|
2016-08-15 13:29:24 +00:00
|
|
|
saddr = nbd_config(s, opts, &export, errp);
|
2015-09-16 13:52:22 +00:00
|
|
|
if (!saddr) {
|
2016-02-10 18:41:12 +00:00
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
|
2016-08-15 13:29:24 +00:00
|
|
|
tlscredsid = g_strdup(qemu_opt_get(opts, "tls-creds"));
|
2016-02-10 18:41:12 +00:00
|
|
|
if (tlscredsid) {
|
|
|
|
tlscreds = nbd_get_tls_creds(tlscredsid, errp);
|
|
|
|
if (!tlscreds) {
|
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (saddr->type != SOCKET_ADDRESS_KIND_INET) {
|
|
|
|
error_setg(errp, "TLS only supported over IP sockets");
|
|
|
|
goto error;
|
|
|
|
}
|
qapi: Don't special-case simple union wrappers
Simple unions were carrying a special case that hid their 'data'
QMP member from the resulting C struct, via the hack method
QAPISchemaObjectTypeVariant.simple_union_type(). But by using
the work we started by unboxing flat union and alternate
branches, coupled with the ability to visit the members of an
implicit type, we can now expose the simple union's implicit
type in qapi-types.h:
| struct q_obj_ImageInfoSpecificQCow2_wrapper {
| ImageInfoSpecificQCow2 *data;
| };
|
| struct q_obj_ImageInfoSpecificVmdk_wrapper {
| ImageInfoSpecificVmdk *data;
| };
...
| struct ImageInfoSpecific {
| ImageInfoSpecificKind type;
| union { /* union tag is @type */
| void *data;
|- ImageInfoSpecificQCow2 *qcow2;
|- ImageInfoSpecificVmdk *vmdk;
|+ q_obj_ImageInfoSpecificQCow2_wrapper qcow2;
|+ q_obj_ImageInfoSpecificVmdk_wrapper vmdk;
| } u;
| };
Doing this removes asymmetry between QAPI's QMP side and its
C side (both sides now expose 'data'), and means that the
treatment of a simple union as sugar for a flat union is now
equivalent in both languages (previously the two approaches used
a different layer of dereferencing, where the simple union could
be converted to a flat union with equivalent C layout but
different {} on the wire, or to an equivalent QMP wire form
but with different C representation). Using the implicit type
also lets us get rid of the simple_union_type() hack.
Of course, now all clients of simple unions have to adjust from
using su->u.member to using su->u.member.data; while this touches
a number of files in the tree, some earlier cleanup patches
helped minimize the change to the initialization of a temporary
variable rather than every single member access. The generated
qapi-visit.c code is also affected by the layout change:
|@@ -7393,10 +7393,10 @@ void visit_type_ImageInfoSpecific_member
| }
| switch (obj->type) {
| case IMAGE_INFO_SPECIFIC_KIND_QCOW2:
|- visit_type_ImageInfoSpecificQCow2(v, "data", &obj->u.qcow2, &err);
|+ visit_type_q_obj_ImageInfoSpecificQCow2_wrapper_members(v, &obj->u.qcow2, &err);
| break;
| case IMAGE_INFO_SPECIFIC_KIND_VMDK:
|- visit_type_ImageInfoSpecificVmdk(v, "data", &obj->u.vmdk, &err);
|+ visit_type_q_obj_ImageInfoSpecificVmdk_wrapper_members(v, &obj->u.vmdk, &err);
| break;
| default:
| abort();
Signed-off-by: Eric Blake <eblake@redhat.com>
Message-Id: <1458254921-17042-13-git-send-email-eblake@redhat.com>
Signed-off-by: Markus Armbruster <armbru@redhat.com>
2016-03-17 22:48:37 +00:00
|
|
|
hostname = saddr->u.inet.data->host;
|
2011-02-22 15:44:54 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/* establish TCP connection, return error if it fails
|
|
|
|
* TODO: Configurable retry-until-timeout behaviour.
|
|
|
|
*/
|
2016-02-10 18:41:01 +00:00
|
|
|
sioc = nbd_establish_connection(saddr, errp);
|
|
|
|
if (!sioc) {
|
2016-02-10 18:41:12 +00:00
|
|
|
ret = -ECONNREFUSED;
|
|
|
|
goto error;
|
2011-10-21 11:16:28 +00:00
|
|
|
}
|
|
|
|
|
2013-12-01 21:23:41 +00:00
|
|
|
/* NBD handshake */
|
2016-02-10 18:41:12 +00:00
|
|
|
ret = nbd_client_init(bs, sioc, export,
|
|
|
|
tlscreds, hostname, errp);
|
|
|
|
error:
|
|
|
|
if (sioc) {
|
|
|
|
object_unref(OBJECT(sioc));
|
|
|
|
}
|
|
|
|
if (tlscreds) {
|
|
|
|
object_unref(OBJECT(tlscreds));
|
|
|
|
}
|
|
|
|
qapi_free_SocketAddress(saddr);
|
2013-12-01 21:23:43 +00:00
|
|
|
g_free(export);
|
2016-08-15 13:29:24 +00:00
|
|
|
qemu_opts_del(opts);
|
2016-02-10 18:41:12 +00:00
|
|
|
return ret;
|
2011-10-20 11:16:23 +00:00
|
|
|
}
|
|
|
|
|
2011-10-21 11:17:14 +00:00
|
|
|
static int nbd_co_flush(BlockDriverState *bs)
|
|
|
|
{
|
2015-02-06 21:06:16 +00:00
|
|
|
return nbd_client_co_flush(bs);
|
2011-10-21 11:17:14 +00:00
|
|
|
}
|
|
|
|
|
2015-02-06 11:24:43 +00:00
|
|
|
static void nbd_refresh_limits(BlockDriverState *bs, Error **errp)
|
|
|
|
{
|
2016-06-23 22:37:21 +00:00
|
|
|
bs->bl.max_pdiscard = NBD_MAX_BUFFER_SIZE;
|
2016-06-23 22:37:19 +00:00
|
|
|
bs->bl.max_transfer = NBD_MAX_BUFFER_SIZE;
|
2015-02-06 11:24:43 +00:00
|
|
|
}
|
|
|
|
|
2008-07-03 13:41:03 +00:00
|
|
|
static void nbd_close(BlockDriverState *bs)
|
|
|
|
{
|
2015-02-06 21:06:16 +00:00
|
|
|
nbd_client_close(bs);
|
2008-07-03 13:41:03 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static int64_t nbd_getlength(BlockDriverState *bs)
|
|
|
|
{
|
|
|
|
BDRVNBDState *s = bs->opaque;
|
|
|
|
|
2013-12-01 21:23:41 +00:00
|
|
|
return s->client.size;
|
2008-07-03 13:41:03 +00:00
|
|
|
}
|
|
|
|
|
2014-05-08 14:34:43 +00:00
|
|
|
static void nbd_detach_aio_context(BlockDriverState *bs)
|
|
|
|
{
|
2015-02-06 21:06:16 +00:00
|
|
|
nbd_client_detach_aio_context(bs);
|
2014-05-08 14:34:43 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static void nbd_attach_aio_context(BlockDriverState *bs,
|
|
|
|
AioContext *new_context)
|
|
|
|
{
|
2015-02-06 21:06:16 +00:00
|
|
|
nbd_client_attach_aio_context(bs, new_context);
|
2014-05-08 14:34:43 +00:00
|
|
|
}
|
|
|
|
|
2015-04-27 11:50:54 +00:00
|
|
|
static void nbd_refresh_filename(BlockDriverState *bs, QDict *options)
|
2014-07-18 18:24:59 +00:00
|
|
|
{
|
|
|
|
QDict *opts = qdict_new();
|
2015-04-27 11:50:54 +00:00
|
|
|
const char *path = qdict_get_try_str(options, "path");
|
|
|
|
const char *host = qdict_get_try_str(options, "host");
|
|
|
|
const char *port = qdict_get_try_str(options, "port");
|
|
|
|
const char *export = qdict_get_try_str(options, "export");
|
2016-02-10 18:41:12 +00:00
|
|
|
const char *tlscreds = qdict_get_try_str(options, "tls-creds");
|
2014-07-18 18:24:59 +00:00
|
|
|
|
|
|
|
qdict_put_obj(opts, "driver", QOBJECT(qstring_from_str("nbd")));
|
|
|
|
|
nbd: Fix filename generation
Export names may be used with nbd+unix, too, fix nbd_refresh_filename()
accordingly. Also, for nbd+tcp, the documented path schema is
"nbd://host[:port]/export", so use it. Furthermore, as can be seen from
that schema, the port is optional.
That makes six single cases for how the filename can be formatted; it is
not easy to generalize these cases without the resulting statement being
completely unreadable, thus there is simply one snprintf() per case.
Finally, taking the options from BDRVNBDState::socket_opts is wrong,
because those will not contain the export name. Just use
BlockDriverState::options instead.
Reported-by: Stefan Hajnoczi <stefanha@redhat.com>
Signed-off-by: Max Reitz <mreitz@redhat.com>
Reviewed-by: Paolo Bonzini <pbonzini@redhat.com>
Signed-off-by: Kevin Wolf <kwolf@redhat.com>
2014-10-08 17:55:15 +00:00
|
|
|
if (path && export) {
|
2014-07-18 18:24:59 +00:00
|
|
|
snprintf(bs->exact_filename, sizeof(bs->exact_filename),
|
nbd: Fix filename generation
Export names may be used with nbd+unix, too, fix nbd_refresh_filename()
accordingly. Also, for nbd+tcp, the documented path schema is
"nbd://host[:port]/export", so use it. Furthermore, as can be seen from
that schema, the port is optional.
That makes six single cases for how the filename can be formatted; it is
not easy to generalize these cases without the resulting statement being
completely unreadable, thus there is simply one snprintf() per case.
Finally, taking the options from BDRVNBDState::socket_opts is wrong,
because those will not contain the export name. Just use
BlockDriverState::options instead.
Reported-by: Stefan Hajnoczi <stefanha@redhat.com>
Signed-off-by: Max Reitz <mreitz@redhat.com>
Reviewed-by: Paolo Bonzini <pbonzini@redhat.com>
Signed-off-by: Kevin Wolf <kwolf@redhat.com>
2014-10-08 17:55:15 +00:00
|
|
|
"nbd+unix:///%s?socket=%s", export, path);
|
|
|
|
} else if (path && !export) {
|
2014-07-18 18:24:59 +00:00
|
|
|
snprintf(bs->exact_filename, sizeof(bs->exact_filename),
|
nbd: Fix filename generation
Export names may be used with nbd+unix, too, fix nbd_refresh_filename()
accordingly. Also, for nbd+tcp, the documented path schema is
"nbd://host[:port]/export", so use it. Furthermore, as can be seen from
that schema, the port is optional.
That makes six single cases for how the filename can be formatted; it is
not easy to generalize these cases without the resulting statement being
completely unreadable, thus there is simply one snprintf() per case.
Finally, taking the options from BDRVNBDState::socket_opts is wrong,
because those will not contain the export name. Just use
BlockDriverState::options instead.
Reported-by: Stefan Hajnoczi <stefanha@redhat.com>
Signed-off-by: Max Reitz <mreitz@redhat.com>
Reviewed-by: Paolo Bonzini <pbonzini@redhat.com>
Signed-off-by: Kevin Wolf <kwolf@redhat.com>
2014-10-08 17:55:15 +00:00
|
|
|
"nbd+unix://?socket=%s", path);
|
|
|
|
} else if (!path && export && port) {
|
|
|
|
snprintf(bs->exact_filename, sizeof(bs->exact_filename),
|
|
|
|
"nbd://%s:%s/%s", host, port, export);
|
|
|
|
} else if (!path && export && !port) {
|
|
|
|
snprintf(bs->exact_filename, sizeof(bs->exact_filename),
|
|
|
|
"nbd://%s/%s", host, export);
|
|
|
|
} else if (!path && !export && port) {
|
|
|
|
snprintf(bs->exact_filename, sizeof(bs->exact_filename),
|
|
|
|
"nbd://%s:%s", host, port);
|
|
|
|
} else if (!path && !export && !port) {
|
2014-07-18 18:24:59 +00:00
|
|
|
snprintf(bs->exact_filename, sizeof(bs->exact_filename),
|
nbd: Fix filename generation
Export names may be used with nbd+unix, too, fix nbd_refresh_filename()
accordingly. Also, for nbd+tcp, the documented path schema is
"nbd://host[:port]/export", so use it. Furthermore, as can be seen from
that schema, the port is optional.
That makes six single cases for how the filename can be formatted; it is
not easy to generalize these cases without the resulting statement being
completely unreadable, thus there is simply one snprintf() per case.
Finally, taking the options from BDRVNBDState::socket_opts is wrong,
because those will not contain the export name. Just use
BlockDriverState::options instead.
Reported-by: Stefan Hajnoczi <stefanha@redhat.com>
Signed-off-by: Max Reitz <mreitz@redhat.com>
Reviewed-by: Paolo Bonzini <pbonzini@redhat.com>
Signed-off-by: Kevin Wolf <kwolf@redhat.com>
2014-10-08 17:55:15 +00:00
|
|
|
"nbd://%s", host);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (path) {
|
|
|
|
qdict_put_obj(opts, "path", QOBJECT(qstring_from_str(path)));
|
|
|
|
} else if (port) {
|
2014-07-18 18:24:59 +00:00
|
|
|
qdict_put_obj(opts, "host", QOBJECT(qstring_from_str(host)));
|
|
|
|
qdict_put_obj(opts, "port", QOBJECT(qstring_from_str(port)));
|
nbd: Fix filename generation
Export names may be used with nbd+unix, too, fix nbd_refresh_filename()
accordingly. Also, for nbd+tcp, the documented path schema is
"nbd://host[:port]/export", so use it. Furthermore, as can be seen from
that schema, the port is optional.
That makes six single cases for how the filename can be formatted; it is
not easy to generalize these cases without the resulting statement being
completely unreadable, thus there is simply one snprintf() per case.
Finally, taking the options from BDRVNBDState::socket_opts is wrong,
because those will not contain the export name. Just use
BlockDriverState::options instead.
Reported-by: Stefan Hajnoczi <stefanha@redhat.com>
Signed-off-by: Max Reitz <mreitz@redhat.com>
Reviewed-by: Paolo Bonzini <pbonzini@redhat.com>
Signed-off-by: Kevin Wolf <kwolf@redhat.com>
2014-10-08 17:55:15 +00:00
|
|
|
} else {
|
|
|
|
qdict_put_obj(opts, "host", QOBJECT(qstring_from_str(host)));
|
|
|
|
}
|
|
|
|
if (export) {
|
|
|
|
qdict_put_obj(opts, "export", QOBJECT(qstring_from_str(export)));
|
2014-07-18 18:24:59 +00:00
|
|
|
}
|
2016-02-10 18:41:12 +00:00
|
|
|
if (tlscreds) {
|
|
|
|
qdict_put_obj(opts, "tls-creds", QOBJECT(qstring_from_str(tlscreds)));
|
|
|
|
}
|
2014-07-18 18:24:59 +00:00
|
|
|
|
|
|
|
bs->full_open_options = opts;
|
|
|
|
}
|
|
|
|
|
2009-05-09 22:03:42 +00:00
|
|
|
static BlockDriver bdrv_nbd = {
|
2014-05-08 14:34:43 +00:00
|
|
|
.format_name = "nbd",
|
|
|
|
.protocol_name = "nbd",
|
|
|
|
.instance_size = sizeof(BDRVNBDState),
|
|
|
|
.bdrv_parse_filename = nbd_parse_filename,
|
|
|
|
.bdrv_file_open = nbd_open,
|
2016-07-15 23:23:07 +00:00
|
|
|
.bdrv_co_preadv = nbd_client_co_preadv,
|
|
|
|
.bdrv_co_pwritev = nbd_client_co_pwritev,
|
2014-05-08 14:34:43 +00:00
|
|
|
.bdrv_close = nbd_close,
|
|
|
|
.bdrv_co_flush_to_os = nbd_co_flush,
|
2016-07-15 23:23:02 +00:00
|
|
|
.bdrv_co_pdiscard = nbd_client_co_pdiscard,
|
2015-02-06 11:24:43 +00:00
|
|
|
.bdrv_refresh_limits = nbd_refresh_limits,
|
2014-05-08 14:34:43 +00:00
|
|
|
.bdrv_getlength = nbd_getlength,
|
|
|
|
.bdrv_detach_aio_context = nbd_detach_aio_context,
|
|
|
|
.bdrv_attach_aio_context = nbd_attach_aio_context,
|
2014-07-18 18:24:59 +00:00
|
|
|
.bdrv_refresh_filename = nbd_refresh_filename,
|
2012-11-04 12:04:24 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
static BlockDriver bdrv_nbd_tcp = {
|
2014-05-08 14:34:43 +00:00
|
|
|
.format_name = "nbd",
|
|
|
|
.protocol_name = "nbd+tcp",
|
|
|
|
.instance_size = sizeof(BDRVNBDState),
|
|
|
|
.bdrv_parse_filename = nbd_parse_filename,
|
|
|
|
.bdrv_file_open = nbd_open,
|
2016-07-15 23:23:07 +00:00
|
|
|
.bdrv_co_preadv = nbd_client_co_preadv,
|
|
|
|
.bdrv_co_pwritev = nbd_client_co_pwritev,
|
2014-05-08 14:34:43 +00:00
|
|
|
.bdrv_close = nbd_close,
|
|
|
|
.bdrv_co_flush_to_os = nbd_co_flush,
|
2016-07-15 23:23:02 +00:00
|
|
|
.bdrv_co_pdiscard = nbd_client_co_pdiscard,
|
2015-02-06 11:24:43 +00:00
|
|
|
.bdrv_refresh_limits = nbd_refresh_limits,
|
2014-05-08 14:34:43 +00:00
|
|
|
.bdrv_getlength = nbd_getlength,
|
|
|
|
.bdrv_detach_aio_context = nbd_detach_aio_context,
|
|
|
|
.bdrv_attach_aio_context = nbd_attach_aio_context,
|
2014-07-18 18:24:59 +00:00
|
|
|
.bdrv_refresh_filename = nbd_refresh_filename,
|
2012-11-04 12:04:24 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
static BlockDriver bdrv_nbd_unix = {
|
2014-05-08 14:34:43 +00:00
|
|
|
.format_name = "nbd",
|
|
|
|
.protocol_name = "nbd+unix",
|
|
|
|
.instance_size = sizeof(BDRVNBDState),
|
|
|
|
.bdrv_parse_filename = nbd_parse_filename,
|
|
|
|
.bdrv_file_open = nbd_open,
|
2016-07-15 23:23:07 +00:00
|
|
|
.bdrv_co_preadv = nbd_client_co_preadv,
|
|
|
|
.bdrv_co_pwritev = nbd_client_co_pwritev,
|
2014-05-08 14:34:43 +00:00
|
|
|
.bdrv_close = nbd_close,
|
|
|
|
.bdrv_co_flush_to_os = nbd_co_flush,
|
2016-07-15 23:23:02 +00:00
|
|
|
.bdrv_co_pdiscard = nbd_client_co_pdiscard,
|
2015-02-06 11:24:43 +00:00
|
|
|
.bdrv_refresh_limits = nbd_refresh_limits,
|
2014-05-08 14:34:43 +00:00
|
|
|
.bdrv_getlength = nbd_getlength,
|
|
|
|
.bdrv_detach_aio_context = nbd_detach_aio_context,
|
|
|
|
.bdrv_attach_aio_context = nbd_attach_aio_context,
|
2014-07-18 18:24:59 +00:00
|
|
|
.bdrv_refresh_filename = nbd_refresh_filename,
|
2008-07-03 13:41:03 +00:00
|
|
|
};
|
2009-05-09 22:03:42 +00:00
|
|
|
|
|
|
|
static void bdrv_nbd_init(void)
|
|
|
|
{
|
|
|
|
bdrv_register(&bdrv_nbd);
|
2012-11-04 12:04:24 +00:00
|
|
|
bdrv_register(&bdrv_nbd_tcp);
|
|
|
|
bdrv_register(&bdrv_nbd_unix);
|
2009-05-09 22:03:42 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
block_init(bdrv_nbd_init);
|