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"
|
2016-10-25 13:11:34 +00:00
|
|
|
#include "qapi-visit.h"
|
|
|
|
#include "qapi/qobject-input-visitor.h"
|
|
|
|
#include "qapi/qobject-output-visitor.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"
|
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 {
|
2016-10-14 18:33:06 +00:00
|
|
|
NBDClientSession client;
|
2016-08-15 13:29:26 +00:00
|
|
|
|
|
|
|
/* For nbd_refresh_filename() */
|
2017-04-26 07:36:40 +00:00
|
|
|
SocketAddress *saddr;
|
2016-10-25 13:11:34 +00:00
|
|
|
char *export, *tlscredsid;
|
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 */
|
2017-06-13 20:57:26 +00:00
|
|
|
if (!g_strcmp0(uri->scheme, "nbd")) {
|
2013-03-07 15:15:11 +00:00
|
|
|
is_unix = false;
|
2017-06-13 20:57:26 +00:00
|
|
|
} else if (!g_strcmp0(uri->scheme, "nbd+tcp")) {
|
2013-03-07 15:15:11 +00:00
|
|
|
is_unix = false;
|
2017-06-13 20:57:26 +00:00
|
|
|
} else if (!g_strcmp0(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]) {
|
2017-04-27 21:58:17 +00:00
|
|
|
qdict_put_str(options, "export", 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;
|
|
|
|
}
|
2017-04-27 21:58:17 +00:00
|
|
|
qdict_put_str(options, "server.type", "unix");
|
|
|
|
qdict_put_str(options, "server.path", qp->p[0].value);
|
2012-11-04 12:04:24 +00:00
|
|
|
} else {
|
2013-06-03 15:54:56 +00:00
|
|
|
QString *host;
|
2016-10-25 13:11:35 +00:00
|
|
|
char *port_str;
|
|
|
|
|
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);
|
|
|
|
}
|
|
|
|
|
2017-04-27 21:58:17 +00:00
|
|
|
qdict_put_str(options, "server.type", "inet");
|
nbd: Tidy up blockdev-add interface
SocketAddress is a simple union, and simple unions are awkward: they
have their variant members wrapped in a "data" object on the wire, and
require additional indirections in C. I intend to limit its use to
existing external interfaces, and convert all internal interfaces to
SocketAddressFlat.
BlockdevOptionsNbd is an external interface using SocketAddress. We
already use SocketAddressFlat elsewhere in blockdev-add. Replace it
by SocketAddressFlat while we can (it's new in 2.9) for simplicity and
consistency. For example,
{ "execute": "blockdev-add",
"arguments": { "node-name": "foo", "driver": "nbd",
"server": { "type": "inet",
"data": { "host": "localhost",
"port": "12345" } } } }
becomes
{ "execute": "blockdev-add",
"arguments": { "node-name": "foo", "driver": "nbd",
"server": { "type": "inet",
"host": "localhost", "port": "12345" } } }
Since the internal interfaces still take SocketAddress, this requires
conversion function socket_address_crumple(). It'll go away when I
update the interfaces.
Unfortunately, SocketAddress is also visible in -drive since 2.8:
-drive if=none,driver=nbd,server.type=inet,server.data.host=127.0.0.1,server.data.port=12345
Nobody should be using it, as it's fairly new and has never been
documented, so adding still more compatibility gunk to keep it working
isn't worth the trouble. You now have to use
-drive if=none,driver=nbd,server.type=inet,server.host=127.0.0.1,server.port=12345
Signed-off-by: Markus Armbruster <armbru@redhat.com>
Message-id: 1490895797-29094-9-git-send-email-armbru@redhat.com
[mreitz: Change iotest 147 accordingly]
Because of this interface change, iotest 147 has to be adapted.
Unfortunately, we cannot just flatten all of the addresses because
nbd-server-start still takes a plain SocketAddress. Therefore, we need
both and this is most easily achieved by writing the SocketAddress into
the code and flattening it where necessary.
Signed-off-by: Max Reitz <mreitz@redhat.com>
Message-id: 20170330221243.17333-1-mreitz@redhat.com
Reviewed-by: Eric Blake <eblake@redhat.com>
Signed-off-by: Max Reitz <mreitz@redhat.com>
2017-03-30 17:43:16 +00:00
|
|
|
qdict_put(options, "server.host", host);
|
2016-10-25 13:11:35 +00:00
|
|
|
|
|
|
|
port_str = g_strdup_printf("%d", uri->port ?: NBD_DEFAULT_PORT);
|
2017-04-27 21:58:17 +00:00
|
|
|
qdict_put_str(options, "server.port", port_str);
|
2016-10-25 13:11:35 +00:00
|
|
|
g_free(port_str);
|
2012-11-04 12:04:24 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
out:
|
|
|
|
if (qp) {
|
|
|
|
query_params_free(qp);
|
|
|
|
}
|
|
|
|
uri_free(uri);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2016-10-25 13:11:33 +00:00
|
|
|
static bool nbd_has_filename_options_conflict(QDict *options, Error **errp)
|
|
|
|
{
|
|
|
|
const QDictEntry *e;
|
|
|
|
|
|
|
|
for (e = qdict_first(options); e; e = qdict_next(options, e)) {
|
|
|
|
if (!strcmp(e->key, "host") ||
|
|
|
|
!strcmp(e->key, "port") ||
|
|
|
|
!strcmp(e->key, "path") ||
|
2016-10-25 13:11:34 +00:00
|
|
|
!strcmp(e->key, "export") ||
|
|
|
|
strstart(e->key, "server.", NULL))
|
2016-10-25 13:11:33 +00:00
|
|
|
{
|
|
|
|
error_setg(errp, "Option '%s' cannot be used with a file name",
|
|
|
|
e->key);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
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;
|
|
|
|
|
2016-10-25 13:11:33 +00:00
|
|
|
if (nbd_has_filename_options_conflict(options, errp)) {
|
2013-03-20 18:23:23 +00:00
|
|
|
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
|
|
|
|
2017-04-27 21:58:17 +00:00
|
|
|
qdict_put_str(options, "export", 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)) {
|
2017-04-27 21:58:17 +00:00
|
|
|
qdict_put_str(options, "server.type", "unix");
|
|
|
|
qdict_put_str(options, "server.path", unixpath);
|
2008-07-03 13:41:03 +00:00
|
|
|
} else {
|
2017-04-26 07:36:37 +00:00
|
|
|
InetSocketAddress *addr = g_new(InetSocketAddress, 1);
|
2013-03-07 15:15:11 +00:00
|
|
|
|
2017-04-26 07:36:37 +00:00
|
|
|
if (inet_parse(addr, host_spec, errp)) {
|
|
|
|
goto out_inet;
|
2013-03-15 10:55:29 +00:00
|
|
|
}
|
2008-07-03 13:41:03 +00:00
|
|
|
|
2017-04-27 21:58:17 +00:00
|
|
|
qdict_put_str(options, "server.type", "inet");
|
|
|
|
qdict_put_str(options, "server.host", addr->host);
|
|
|
|
qdict_put_str(options, "server.port", addr->port);
|
2017-04-26 07:36:37 +00:00
|
|
|
out_inet:
|
2013-03-07 15:15:11 +00:00
|
|
|
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-10-25 13:11:34 +00:00
|
|
|
static bool nbd_process_legacy_socket_options(QDict *output_options,
|
|
|
|
QemuOpts *legacy_opts,
|
|
|
|
Error **errp)
|
2013-03-07 15:15:11 +00:00
|
|
|
{
|
2016-10-25 13:11:34 +00:00
|
|
|
const char *path = qemu_opt_get(legacy_opts, "path");
|
|
|
|
const char *host = qemu_opt_get(legacy_opts, "host");
|
|
|
|
const char *port = qemu_opt_get(legacy_opts, "port");
|
|
|
|
const QDictEntry *e;
|
2013-03-07 15:15:11 +00:00
|
|
|
|
2016-10-25 13:11:34 +00:00
|
|
|
if (!path && !host && !port) {
|
|
|
|
return true;
|
|
|
|
}
|
2016-08-15 13:29:26 +00:00
|
|
|
|
2016-10-25 13:11:34 +00:00
|
|
|
for (e = qdict_first(output_options); e; e = qdict_next(output_options, e))
|
|
|
|
{
|
|
|
|
if (strstart(e->key, "server.", NULL)) {
|
|
|
|
error_setg(errp, "Cannot use 'server' and path/host/port at the "
|
|
|
|
"same time");
|
|
|
|
return false;
|
2013-03-20 18:23:23 +00:00
|
|
|
}
|
2011-02-22 15:44:54 +00:00
|
|
|
}
|
2016-10-25 13:11:34 +00:00
|
|
|
|
|
|
|
if (path && host) {
|
|
|
|
error_setg(errp, "path and host may not be used at the same time");
|
|
|
|
return false;
|
|
|
|
} else if (path) {
|
|
|
|
if (port) {
|
|
|
|
error_setg(errp, "port may not be used without host");
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2017-04-27 21:58:17 +00:00
|
|
|
qdict_put_str(output_options, "server.type", "unix");
|
|
|
|
qdict_put_str(output_options, "server.path", path);
|
2016-10-25 13:11:34 +00:00
|
|
|
} else if (host) {
|
2017-04-27 21:58:17 +00:00
|
|
|
qdict_put_str(output_options, "server.type", "inet");
|
|
|
|
qdict_put_str(output_options, "server.host", host);
|
|
|
|
qdict_put_str(output_options, "server.port",
|
|
|
|
port ?: stringify(NBD_DEFAULT_PORT));
|
2016-10-25 13:11:30 +00:00
|
|
|
}
|
2013-03-07 15:15:11 +00:00
|
|
|
|
2016-10-25 13:11:34 +00:00
|
|
|
return true;
|
|
|
|
}
|
2013-03-07 15:15:11 +00:00
|
|
|
|
2017-04-26 07:36:40 +00:00
|
|
|
static SocketAddress *nbd_config(BDRVNBDState *s, QDict *options,
|
|
|
|
Error **errp)
|
2016-10-25 13:11:34 +00:00
|
|
|
{
|
2017-04-26 07:36:40 +00:00
|
|
|
SocketAddress *saddr = NULL;
|
2016-10-25 13:11:34 +00:00
|
|
|
QDict *addr = NULL;
|
|
|
|
QObject *crumpled_addr = NULL;
|
|
|
|
Visitor *iv = NULL;
|
|
|
|
Error *local_err = NULL;
|
|
|
|
|
|
|
|
qdict_extract_subqdict(options, &addr, "server.");
|
|
|
|
if (!qdict_size(addr)) {
|
|
|
|
error_setg(errp, "NBD server address missing");
|
|
|
|
goto done;
|
2013-03-07 15:15:11 +00:00
|
|
|
}
|
|
|
|
|
2016-10-25 13:11:34 +00:00
|
|
|
crumpled_addr = qdict_crumple(addr, errp);
|
|
|
|
if (!crumpled_addr) {
|
|
|
|
goto done;
|
|
|
|
}
|
2013-03-18 15:56:05 +00:00
|
|
|
|
block: Document -drive problematic code and bugs
-blockdev and blockdev_add convert their arguments via QObject to
BlockdevOptions for qmp_blockdev_add(), which converts them back to
QObject, then to a flattened QDict. The QDict's members are typed
according to the QAPI schema.
-drive converts its argument via QemuOpts to a (flat) QDict. This
QDict's members are all QString.
Thus, the QType of a flat QDict member depends on whether it comes
from -drive or -blockdev/blockdev_add, except when the QAPI type maps
to QString, which is the case for 'str' and enumeration types.
The block layer core extracts generic configuration from the flat
QDict, and the block driver extracts driver-specific configuration.
Both commonly do so by converting (parts of) the flat QDict to
QemuOpts, which turns all values into strings. Not exactly elegant,
but correct.
However, A few places access the flat QDict directly:
* Most of them access members that are always QString. Correct.
* bdrv_open_inherit() accesses a boolean, carefully. Correct.
* nfs_config() uses a QObject input visitor. Correct only because the
visited type contains nothing but QStrings.
* nbd_config() and ssh_config() use a QObject input visitor, and the
visited types contain non-QStrings: InetSocketAddress members
@numeric, @to, @ipv4, @ipv6. -drive works as long as you don't try
to use them (they're all optional). @to is ignored anyway.
Reproducer:
-drive driver=ssh,server.host=h,server.port=22,server.ipv4,path=p
-drive driver=nbd,server.type=inet,server.data.host=h,server.data.port=22,server.data.ipv4
both fail with "Invalid parameter type for 'data.ipv4', expected: boolean"
Add suitable comments to all these places. Mark the buggy ones FIXME.
"Fortunately", -drive's driver-specific options are entirely
undocumented.
Signed-off-by: Markus Armbruster <armbru@redhat.com>
Message-id: 1490895797-29094-5-git-send-email-armbru@redhat.com
[mreitz: Fixed two typos]
Reviewed-by: Eric Blake <eblake@redhat.com>
Signed-off-by: Max Reitz <mreitz@redhat.com>
2017-03-30 17:43:12 +00:00
|
|
|
/*
|
|
|
|
* FIXME .numeric, .to, .ipv4 or .ipv6 don't work with -drive
|
|
|
|
* server.type=inet. .to doesn't matter, it's ignored anyway.
|
|
|
|
* That's because when @options come from -blockdev or
|
|
|
|
* blockdev_add, members are typed according to the QAPI schema,
|
|
|
|
* but when they come from -drive, they're all QString. The
|
|
|
|
* visitor expects the former.
|
|
|
|
*/
|
2017-03-03 12:32:39 +00:00
|
|
|
iv = qobject_input_visitor_new(crumpled_addr);
|
2017-04-26 07:36:40 +00:00
|
|
|
visit_type_SocketAddress(iv, NULL, &saddr, &local_err);
|
2016-10-25 13:11:34 +00:00
|
|
|
if (local_err) {
|
|
|
|
error_propagate(errp, local_err);
|
|
|
|
goto done;
|
|
|
|
}
|
2015-09-16 13:52:22 +00:00
|
|
|
|
2016-10-25 13:11:34 +00:00
|
|
|
done:
|
|
|
|
QDECREF(addr);
|
|
|
|
qobject_decref(crumpled_addr);
|
|
|
|
visit_free(iv);
|
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
|
|
|
|
2016-10-14 18:33:06 +00:00
|
|
|
NBDClientSession *nbd_get_client_session(BlockDriverState *bs)
|
2015-02-06 21:06:16 +00:00
|
|
|
{
|
|
|
|
BDRVNBDState *s = bs->opaque;
|
|
|
|
return &s->client;
|
|
|
|
}
|
|
|
|
|
2017-04-26 07:36:41 +00:00
|
|
|
static QIOChannelSocket *nbd_establish_connection(SocketAddress *saddr,
|
2016-02-10 18:41:01 +00:00
|
|
|
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();
|
2016-09-30 10:57:14 +00:00
|
|
|
qio_channel_set_name(QIO_CHANNEL(sioc), "nbd-client");
|
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) {
|
2017-04-01 00:15:09 +00:00
|
|
|
object_unref(OBJECT(sioc));
|
2016-02-10 18:41:01 +00:00
|
|
|
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;
|
2016-02-10 18:41:12 +00:00
|
|
|
QIOChannelSocket *sioc = NULL;
|
|
|
|
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;
|
|
|
|
}
|
|
|
|
|
2017-04-26 07:36:40 +00:00
|
|
|
/* Translate @host, @port, and @path to a SocketAddress */
|
2016-10-25 13:11:34 +00:00
|
|
|
if (!nbd_process_legacy_socket_options(options, opts, errp)) {
|
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
|
2011-02-22 15:44:54 +00:00
|
|
|
/* Pop the config into our state object. Exit if invalid. */
|
2016-10-25 13:11:34 +00:00
|
|
|
s->saddr = nbd_config(s, options, errp);
|
|
|
|
if (!s->saddr) {
|
2016-02-10 18:41:12 +00:00
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
|
2016-10-25 13:11:34 +00:00
|
|
|
s->export = g_strdup(qemu_opt_get(opts, "export"));
|
|
|
|
|
2016-08-15 13:29:26 +00:00
|
|
|
s->tlscredsid = g_strdup(qemu_opt_get(opts, "tls-creds"));
|
|
|
|
if (s->tlscredsid) {
|
|
|
|
tlscreds = nbd_get_tls_creds(s->tlscredsid, errp);
|
2016-02-10 18:41:12 +00:00
|
|
|
if (!tlscreds) {
|
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
|
2017-03-30 17:43:09 +00:00
|
|
|
/* TODO SOCKET_ADDRESS_KIND_FD where fd has AF_INET or AF_INET6 */
|
2017-04-26 07:36:40 +00:00
|
|
|
if (s->saddr->type != SOCKET_ADDRESS_TYPE_INET) {
|
2016-02-10 18:41:12 +00:00
|
|
|
error_setg(errp, "TLS only supported over IP sockets");
|
|
|
|
goto error;
|
|
|
|
}
|
nbd: Tidy up blockdev-add interface
SocketAddress is a simple union, and simple unions are awkward: they
have their variant members wrapped in a "data" object on the wire, and
require additional indirections in C. I intend to limit its use to
existing external interfaces, and convert all internal interfaces to
SocketAddressFlat.
BlockdevOptionsNbd is an external interface using SocketAddress. We
already use SocketAddressFlat elsewhere in blockdev-add. Replace it
by SocketAddressFlat while we can (it's new in 2.9) for simplicity and
consistency. For example,
{ "execute": "blockdev-add",
"arguments": { "node-name": "foo", "driver": "nbd",
"server": { "type": "inet",
"data": { "host": "localhost",
"port": "12345" } } } }
becomes
{ "execute": "blockdev-add",
"arguments": { "node-name": "foo", "driver": "nbd",
"server": { "type": "inet",
"host": "localhost", "port": "12345" } } }
Since the internal interfaces still take SocketAddress, this requires
conversion function socket_address_crumple(). It'll go away when I
update the interfaces.
Unfortunately, SocketAddress is also visible in -drive since 2.8:
-drive if=none,driver=nbd,server.type=inet,server.data.host=127.0.0.1,server.data.port=12345
Nobody should be using it, as it's fairly new and has never been
documented, so adding still more compatibility gunk to keep it working
isn't worth the trouble. You now have to use
-drive if=none,driver=nbd,server.type=inet,server.host=127.0.0.1,server.port=12345
Signed-off-by: Markus Armbruster <armbru@redhat.com>
Message-id: 1490895797-29094-9-git-send-email-armbru@redhat.com
[mreitz: Change iotest 147 accordingly]
Because of this interface change, iotest 147 has to be adapted.
Unfortunately, we cannot just flatten all of the addresses because
nbd-server-start still takes a plain SocketAddress. Therefore, we need
both and this is most easily achieved by writing the SocketAddress into
the code and flattening it where necessary.
Signed-off-by: Max Reitz <mreitz@redhat.com>
Message-id: 20170330221243.17333-1-mreitz@redhat.com
Reviewed-by: Eric Blake <eblake@redhat.com>
Signed-off-by: Max Reitz <mreitz@redhat.com>
2017-03-30 17:43:16 +00:00
|
|
|
hostname = s->saddr->u.inet.host;
|
2011-02-22 15:44:54 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/* establish TCP connection, return error if it fails
|
|
|
|
* TODO: Configurable retry-until-timeout behaviour.
|
|
|
|
*/
|
2016-10-25 13:11:34 +00:00
|
|
|
sioc = nbd_establish_connection(s->saddr, errp);
|
2016-02-10 18:41:01 +00:00
|
|
|
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-08-15 13:29:26 +00:00
|
|
|
ret = nbd_client_init(bs, sioc, s->export,
|
2016-02-10 18:41:12 +00:00
|
|
|
tlscreds, hostname, errp);
|
|
|
|
error:
|
|
|
|
if (sioc) {
|
|
|
|
object_unref(OBJECT(sioc));
|
|
|
|
}
|
|
|
|
if (tlscreds) {
|
|
|
|
object_unref(OBJECT(tlscreds));
|
|
|
|
}
|
2016-08-15 13:29:26 +00:00
|
|
|
if (ret < 0) {
|
2017-04-26 07:36:40 +00:00
|
|
|
qapi_free_SocketAddress(s->saddr);
|
2016-08-15 13:29:26 +00:00
|
|
|
g_free(s->export);
|
|
|
|
g_free(s->tlscredsid);
|
|
|
|
}
|
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)
|
|
|
|
{
|
nbd: Implement NBD_INFO_BLOCK_SIZE on client
The upstream NBD Protocol has defined a new extension to allow
the server to advertise block sizes to the client, as well as
a way for the client to inform the server whether it intends to
obey block sizes.
When using the block layer as the client, we will obey block
sizes; but when used as 'qemu-nbd -c' to hand off to the
kernel nbd module as the client, we are still waiting for the
kernel to implement a way for us to learn if it will honor
block sizes (perhaps by an addition to sysfs, rather than an
ioctl), as well as any way to tell the kernel what additional
block sizes to obey (NBD_SET_BLKSIZE appears to be accurate
for the minimum size, but preferred and maximum sizes would
probably be new ioctl()s), so until then, we need to make our
request for block sizes conditional.
When using ioctl(NBD_SET_BLKSIZE) to hand off to the kernel,
use the minimum block size as the sector size if it is larger
than 512, which also has the nice effect of cooperating with
(non-qemu) servers that don't do read-modify-write when
exposing a block device with 4k sectors; it might also allow
us to visit a file larger than 2T on a 32-bit kernel.
Signed-off-by: Eric Blake <eblake@redhat.com>
Message-Id: <20170707203049.534-10-eblake@redhat.com>
Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
2017-07-07 20:30:49 +00:00
|
|
|
NBDClientSession *s = nbd_get_client_session(bs);
|
|
|
|
uint32_t max = MIN_NON_ZERO(NBD_MAX_BUFFER_SIZE, s->info.max_block);
|
|
|
|
|
|
|
|
bs->bl.max_pdiscard = max;
|
|
|
|
bs->bl.max_pwrite_zeroes = max;
|
|
|
|
bs->bl.max_transfer = max;
|
|
|
|
|
|
|
|
if (s->info.opt_block &&
|
|
|
|
s->info.opt_block > bs->bl.opt_transfer) {
|
|
|
|
bs->bl.opt_transfer = s->info.opt_block;
|
|
|
|
}
|
2015-02-06 11:24:43 +00:00
|
|
|
}
|
|
|
|
|
2008-07-03 13:41:03 +00:00
|
|
|
static void nbd_close(BlockDriverState *bs)
|
|
|
|
{
|
2016-08-15 13:29:26 +00:00
|
|
|
BDRVNBDState *s = bs->opaque;
|
|
|
|
|
2015-02-06 21:06:16 +00:00
|
|
|
nbd_client_close(bs);
|
2016-08-15 13:29:26 +00:00
|
|
|
|
2017-04-26 07:36:40 +00:00
|
|
|
qapi_free_SocketAddress(s->saddr);
|
2016-08-15 13:29:26 +00:00
|
|
|
g_free(s->export);
|
|
|
|
g_free(s->tlscredsid);
|
2008-07-03 13:41:03 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static int64_t nbd_getlength(BlockDriverState *bs)
|
|
|
|
{
|
|
|
|
BDRVNBDState *s = bs->opaque;
|
|
|
|
|
2017-07-07 20:30:41 +00:00
|
|
|
return s->client.info.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
|
|
|
{
|
2016-08-15 13:29:26 +00:00
|
|
|
BDRVNBDState *s = bs->opaque;
|
2014-07-18 18:24:59 +00:00
|
|
|
QDict *opts = qdict_new();
|
2016-10-25 13:11:34 +00:00
|
|
|
QObject *saddr_qdict;
|
|
|
|
Visitor *ov;
|
|
|
|
const char *host = NULL, *port = NULL, *path = NULL;
|
|
|
|
|
2017-04-26 07:36:40 +00:00
|
|
|
if (s->saddr->type == SOCKET_ADDRESS_TYPE_INET) {
|
nbd: Tidy up blockdev-add interface
SocketAddress is a simple union, and simple unions are awkward: they
have their variant members wrapped in a "data" object on the wire, and
require additional indirections in C. I intend to limit its use to
existing external interfaces, and convert all internal interfaces to
SocketAddressFlat.
BlockdevOptionsNbd is an external interface using SocketAddress. We
already use SocketAddressFlat elsewhere in blockdev-add. Replace it
by SocketAddressFlat while we can (it's new in 2.9) for simplicity and
consistency. For example,
{ "execute": "blockdev-add",
"arguments": { "node-name": "foo", "driver": "nbd",
"server": { "type": "inet",
"data": { "host": "localhost",
"port": "12345" } } } }
becomes
{ "execute": "blockdev-add",
"arguments": { "node-name": "foo", "driver": "nbd",
"server": { "type": "inet",
"host": "localhost", "port": "12345" } } }
Since the internal interfaces still take SocketAddress, this requires
conversion function socket_address_crumple(). It'll go away when I
update the interfaces.
Unfortunately, SocketAddress is also visible in -drive since 2.8:
-drive if=none,driver=nbd,server.type=inet,server.data.host=127.0.0.1,server.data.port=12345
Nobody should be using it, as it's fairly new and has never been
documented, so adding still more compatibility gunk to keep it working
isn't worth the trouble. You now have to use
-drive if=none,driver=nbd,server.type=inet,server.host=127.0.0.1,server.port=12345
Signed-off-by: Markus Armbruster <armbru@redhat.com>
Message-id: 1490895797-29094-9-git-send-email-armbru@redhat.com
[mreitz: Change iotest 147 accordingly]
Because of this interface change, iotest 147 has to be adapted.
Unfortunately, we cannot just flatten all of the addresses because
nbd-server-start still takes a plain SocketAddress. Therefore, we need
both and this is most easily achieved by writing the SocketAddress into
the code and flattening it where necessary.
Signed-off-by: Max Reitz <mreitz@redhat.com>
Message-id: 20170330221243.17333-1-mreitz@redhat.com
Reviewed-by: Eric Blake <eblake@redhat.com>
Signed-off-by: Max Reitz <mreitz@redhat.com>
2017-03-30 17:43:16 +00:00
|
|
|
const InetSocketAddress *inet = &s->saddr->u.inet;
|
2016-10-25 13:11:34 +00:00
|
|
|
if (!inet->has_ipv4 && !inet->has_ipv6 && !inet->has_to) {
|
|
|
|
host = inet->host;
|
|
|
|
port = inet->port;
|
|
|
|
}
|
2017-04-26 07:36:40 +00:00
|
|
|
} else if (s->saddr->type == SOCKET_ADDRESS_TYPE_UNIX) {
|
nbd: Tidy up blockdev-add interface
SocketAddress is a simple union, and simple unions are awkward: they
have their variant members wrapped in a "data" object on the wire, and
require additional indirections in C. I intend to limit its use to
existing external interfaces, and convert all internal interfaces to
SocketAddressFlat.
BlockdevOptionsNbd is an external interface using SocketAddress. We
already use SocketAddressFlat elsewhere in blockdev-add. Replace it
by SocketAddressFlat while we can (it's new in 2.9) for simplicity and
consistency. For example,
{ "execute": "blockdev-add",
"arguments": { "node-name": "foo", "driver": "nbd",
"server": { "type": "inet",
"data": { "host": "localhost",
"port": "12345" } } } }
becomes
{ "execute": "blockdev-add",
"arguments": { "node-name": "foo", "driver": "nbd",
"server": { "type": "inet",
"host": "localhost", "port": "12345" } } }
Since the internal interfaces still take SocketAddress, this requires
conversion function socket_address_crumple(). It'll go away when I
update the interfaces.
Unfortunately, SocketAddress is also visible in -drive since 2.8:
-drive if=none,driver=nbd,server.type=inet,server.data.host=127.0.0.1,server.data.port=12345
Nobody should be using it, as it's fairly new and has never been
documented, so adding still more compatibility gunk to keep it working
isn't worth the trouble. You now have to use
-drive if=none,driver=nbd,server.type=inet,server.host=127.0.0.1,server.port=12345
Signed-off-by: Markus Armbruster <armbru@redhat.com>
Message-id: 1490895797-29094-9-git-send-email-armbru@redhat.com
[mreitz: Change iotest 147 accordingly]
Because of this interface change, iotest 147 has to be adapted.
Unfortunately, we cannot just flatten all of the addresses because
nbd-server-start still takes a plain SocketAddress. Therefore, we need
both and this is most easily achieved by writing the SocketAddress into
the code and flattening it where necessary.
Signed-off-by: Max Reitz <mreitz@redhat.com>
Message-id: 20170330221243.17333-1-mreitz@redhat.com
Reviewed-by: Eric Blake <eblake@redhat.com>
Signed-off-by: Max Reitz <mreitz@redhat.com>
2017-03-30 17:43:16 +00:00
|
|
|
path = s->saddr->u.q_unix.path;
|
|
|
|
} /* else can't represent as pseudo-filename */
|
2014-07-18 18:24:59 +00:00
|
|
|
|
2017-04-27 21:58:17 +00:00
|
|
|
qdict_put_str(opts, "driver", "nbd");
|
2014-07-18 18:24:59 +00:00
|
|
|
|
2016-10-25 13:11:34 +00:00
|
|
|
if (path && s->export) {
|
2014-07-18 18:24:59 +00:00
|
|
|
snprintf(bs->exact_filename, sizeof(bs->exact_filename),
|
2016-10-25 13:11:34 +00:00
|
|
|
"nbd+unix:///%s?socket=%s", s->export, path);
|
|
|
|
} else if (path && !s->export) {
|
2014-07-18 18:24:59 +00:00
|
|
|
snprintf(bs->exact_filename, sizeof(bs->exact_filename),
|
2016-10-25 13:11:34 +00:00
|
|
|
"nbd+unix://?socket=%s", path);
|
|
|
|
} else if (host && s->export) {
|
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
|
|
|
snprintf(bs->exact_filename, sizeof(bs->exact_filename),
|
2016-10-25 13:11:34 +00:00
|
|
|
"nbd://%s:%s/%s", host, port, s->export);
|
|
|
|
} else if (host && !s->export) {
|
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
|
|
|
snprintf(bs->exact_filename, sizeof(bs->exact_filename),
|
2016-10-25 13:11:34 +00:00
|
|
|
"nbd://%s:%s", host, 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
|
|
|
}
|
|
|
|
|
2016-10-25 13:11:34 +00:00
|
|
|
ov = qobject_output_visitor_new(&saddr_qdict);
|
2017-04-26 07:36:40 +00:00
|
|
|
visit_type_SocketAddress(ov, NULL, &s->saddr, &error_abort);
|
2016-10-25 13:11:34 +00:00
|
|
|
visit_complete(ov, &saddr_qdict);
|
2016-11-02 10:40:03 +00:00
|
|
|
visit_free(ov);
|
2016-10-25 13:11:34 +00:00
|
|
|
qdict_put_obj(opts, "server", saddr_qdict);
|
|
|
|
|
2016-08-15 13:29:26 +00:00
|
|
|
if (s->export) {
|
2017-04-27 21:58:17 +00:00
|
|
|
qdict_put_str(opts, "export", s->export);
|
2014-07-18 18:24:59 +00:00
|
|
|
}
|
2016-08-15 13:29:26 +00:00
|
|
|
if (s->tlscredsid) {
|
2017-04-27 21:58:17 +00:00
|
|
|
qdict_put_str(opts, "tls-creds", s->tlscredsid);
|
2016-02-10 18:41:12 +00:00
|
|
|
}
|
2014-07-18 18:24:59 +00:00
|
|
|
|
2016-10-25 13:11:34 +00:00
|
|
|
qdict_flatten(opts);
|
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,
|
2016-10-14 18:33:18 +00:00
|
|
|
.bdrv_co_pwrite_zeroes = nbd_client_co_pwrite_zeroes,
|
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,
|
2016-10-14 18:33:18 +00:00
|
|
|
.bdrv_co_pwrite_zeroes = nbd_client_co_pwrite_zeroes,
|
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,
|
2016-10-14 18:33:18 +00:00
|
|
|
.bdrv_co_pwrite_zeroes = nbd_client_co_pwrite_zeroes,
|
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);
|