mirror of
https://gitlab.com/libvirt/libvirt.git
synced 2025-01-14 08:35:15 +00:00
ae65c908b7
https://bugzilla.redhat.com/show_bug.cgi?id=1356436 According to RFC 3721 (https://www.ietf.org/rfc/rfc3721.txt), there are two ways to "discover" targets in/for the iSCSI environment. Discovery is the process which allows the initiator to find the targets to which it has access and at least one address at which each target may be accessed. The method currently implemented in libvirt using the virISCSIScanTargets API is known as "SendTargets" discovery. This method is more useful when the target IP Address and TCP port information are available, e.g. in libvirt terms the "portal". It returns a list of targets for the portal. From that list, the target can be found. This operation can also fill an iSCSI node table into which iSCSI logins may occur. Commit id '56057900' altered that filling by adding the "--op nonpersistent" since it was not necessarily desired to perform that for non libvirt related targets. The second method is "Static Configuration". This method not only needs the IP Address and TCP port (e.g. portal), but also the iSCSI target name. In libvirt terms this would be the device path field from the iSCSI pool <source> XML. This patch implements the second methodology using that required device path as the targetname.
531 lines
15 KiB
C
531 lines
15 KiB
C
/*
|
|
* viriscsi.c: helper APIs for managing iSCSI
|
|
*
|
|
* Copyright (C) 2007-2014 Red Hat, Inc.
|
|
* Copyright (C) 2007-2008 Daniel P. Berrange
|
|
*
|
|
* 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 <config.h>
|
|
|
|
#include <regex.h>
|
|
#include <stdio.h>
|
|
|
|
#include "viriscsi.h"
|
|
|
|
#include "viralloc.h"
|
|
#include "vircommand.h"
|
|
#include "virerror.h"
|
|
#include "virfile.h"
|
|
#include "virlog.h"
|
|
#include "virrandom.h"
|
|
#include "virstring.h"
|
|
|
|
#define VIR_FROM_THIS VIR_FROM_NONE
|
|
|
|
VIR_LOG_INIT("util.iscsi");
|
|
|
|
|
|
struct virISCSISessionData {
|
|
char *session;
|
|
const char *devpath;
|
|
};
|
|
|
|
|
|
static int
|
|
virISCSIExtractSession(char **const groups,
|
|
void *opaque)
|
|
{
|
|
struct virISCSISessionData *data = opaque;
|
|
|
|
if (STREQ(groups[1], data->devpath))
|
|
return VIR_STRDUP(data->session, groups[0]);
|
|
return 0;
|
|
}
|
|
|
|
|
|
char *
|
|
virISCSIGetSession(const char *devpath,
|
|
bool probe)
|
|
{
|
|
/*
|
|
* # iscsiadm --mode session
|
|
* tcp: [1] 192.168.122.170:3260,1 demo-tgt-b
|
|
* tcp: [2] 192.168.122.170:3260,1 demo-tgt-a
|
|
*
|
|
* Pull out 2nd and 4th fields
|
|
*/
|
|
const char *regexes[] = {
|
|
"^tcp:\\s+\\[(\\S+)\\]\\s+\\S+\\s+(\\S+).*$"
|
|
};
|
|
int vars[] = {
|
|
2,
|
|
};
|
|
struct virISCSISessionData cbdata = {
|
|
.session = NULL,
|
|
.devpath = devpath,
|
|
};
|
|
char *error = NULL;
|
|
int exitstatus = 0;
|
|
|
|
virCommandPtr cmd = virCommandNewArgList(ISCSIADM, "--mode",
|
|
"session", NULL);
|
|
virCommandSetErrorBuffer(cmd, &error);
|
|
|
|
if (virCommandRunRegex(cmd,
|
|
1,
|
|
regexes,
|
|
vars,
|
|
virISCSIExtractSession,
|
|
&cbdata, NULL, &exitstatus) < 0)
|
|
goto cleanup;
|
|
|
|
if (cbdata.session == NULL && !probe)
|
|
virReportError(VIR_ERR_INTERNAL_ERROR,
|
|
_("cannot find iscsiadm session: %s"),
|
|
NULLSTR(error));
|
|
|
|
cleanup:
|
|
VIR_FREE(error);
|
|
virCommandFree(cmd);
|
|
return cbdata.session;
|
|
}
|
|
|
|
|
|
|
|
#define LINE_SIZE 4096
|
|
#define IQN_FOUND 1
|
|
#define IQN_MISSING 0
|
|
#define IQN_ERROR -1
|
|
|
|
static int
|
|
virStorageBackendIQNFound(const char *initiatoriqn,
|
|
char **ifacename)
|
|
{
|
|
int ret = IQN_MISSING, fd = -1;
|
|
char ebuf[64];
|
|
FILE *fp = NULL;
|
|
char *line = NULL, *newline = NULL, *iqn = NULL, *token = NULL;
|
|
virCommandPtr cmd = virCommandNewArgList(ISCSIADM,
|
|
"--mode", "iface", NULL);
|
|
|
|
if (VIR_ALLOC_N(line, LINE_SIZE) != 0) {
|
|
ret = IQN_ERROR;
|
|
virReportError(VIR_ERR_INTERNAL_ERROR,
|
|
_("Could not allocate memory for output of '%s'"),
|
|
ISCSIADM);
|
|
goto out;
|
|
}
|
|
|
|
memset(line, 0, LINE_SIZE);
|
|
|
|
virCommandSetOutputFD(cmd, &fd);
|
|
if (virCommandRunAsync(cmd, NULL) < 0) {
|
|
ret = IQN_ERROR;
|
|
goto out;
|
|
}
|
|
|
|
if ((fp = VIR_FDOPEN(fd, "r")) == NULL) {
|
|
virReportError(VIR_ERR_INTERNAL_ERROR,
|
|
_("Failed to open stream for file descriptor "
|
|
"when reading output from '%s': '%s'"),
|
|
ISCSIADM, virStrerror(errno, ebuf, sizeof(ebuf)));
|
|
ret = IQN_ERROR;
|
|
goto out;
|
|
}
|
|
|
|
while (fgets(line, LINE_SIZE, fp) != NULL) {
|
|
newline = strrchr(line, '\n');
|
|
if (newline == NULL) {
|
|
ret = IQN_ERROR;
|
|
virReportError(VIR_ERR_INTERNAL_ERROR,
|
|
_("Unexpected line > %d characters "
|
|
"when parsing output of '%s'"),
|
|
LINE_SIZE, ISCSIADM);
|
|
goto out;
|
|
}
|
|
*newline = '\0';
|
|
|
|
iqn = strrchr(line, ',');
|
|
if (iqn == NULL)
|
|
continue;
|
|
iqn++;
|
|
|
|
if (STREQ(iqn, initiatoriqn)) {
|
|
token = strchr(line, ' ');
|
|
if (!token) {
|
|
ret = IQN_ERROR;
|
|
virReportError(VIR_ERR_INTERNAL_ERROR,
|
|
_("Missing space when parsing output "
|
|
"of '%s'"), ISCSIADM);
|
|
goto out;
|
|
}
|
|
if (VIR_STRNDUP(*ifacename, line, token - line) < 0) {
|
|
ret = IQN_ERROR;
|
|
goto out;
|
|
}
|
|
VIR_DEBUG("Found interface '%s' with IQN '%s'", *ifacename, iqn);
|
|
ret = IQN_FOUND;
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (virCommandWait(cmd, NULL) < 0)
|
|
ret = IQN_ERROR;
|
|
|
|
out:
|
|
if (ret == IQN_MISSING)
|
|
VIR_DEBUG("Could not find interface with IQN '%s'", iqn);
|
|
|
|
VIR_FREE(line);
|
|
VIR_FORCE_FCLOSE(fp);
|
|
VIR_FORCE_CLOSE(fd);
|
|
virCommandFree(cmd);
|
|
|
|
return ret;
|
|
}
|
|
|
|
|
|
static int
|
|
virStorageBackendCreateIfaceIQN(const char *initiatoriqn,
|
|
char **ifacename)
|
|
{
|
|
int ret = -1, exitstatus = -1;
|
|
char *temp_ifacename;
|
|
virCommandPtr cmd = NULL;
|
|
|
|
if (virAsprintf(&temp_ifacename,
|
|
"libvirt-iface-%08llx",
|
|
(unsigned long long)virRandomBits(30)) < 0)
|
|
return -1;
|
|
|
|
VIR_DEBUG("Attempting to create interface '%s' with IQN '%s'",
|
|
temp_ifacename, initiatoriqn);
|
|
|
|
cmd = virCommandNewArgList(ISCSIADM,
|
|
"--mode", "iface",
|
|
"--interface", temp_ifacename,
|
|
"--op", "new",
|
|
NULL);
|
|
/* Note that we ignore the exitstatus. Older versions of iscsiadm
|
|
* tools returned an exit status of > 0, even if they succeeded.
|
|
* We will just rely on whether the interface got created
|
|
* properly. */
|
|
if (virCommandRun(cmd, &exitstatus) < 0) {
|
|
virReportError(VIR_ERR_INTERNAL_ERROR,
|
|
_("Failed to run command '%s' to create new iscsi interface"),
|
|
ISCSIADM);
|
|
goto cleanup;
|
|
}
|
|
virCommandFree(cmd);
|
|
|
|
cmd = virCommandNewArgList(ISCSIADM,
|
|
"--mode", "iface",
|
|
"--interface", temp_ifacename,
|
|
"--op", "update",
|
|
"--name", "iface.initiatorname",
|
|
"--value",
|
|
initiatoriqn,
|
|
NULL);
|
|
/* Note that we ignore the exitstatus. Older versions of iscsiadm tools
|
|
* returned an exit status of > 0, even if they succeeded. We will just
|
|
* rely on whether iface file got updated properly. */
|
|
if (virCommandRun(cmd, &exitstatus) < 0) {
|
|
virReportError(VIR_ERR_INTERNAL_ERROR,
|
|
_("Failed to run command '%s' to update iscsi interface with IQN '%s'"),
|
|
ISCSIADM, initiatoriqn);
|
|
goto cleanup;
|
|
}
|
|
|
|
/* Check again to make sure the interface was created. */
|
|
if (virStorageBackendIQNFound(initiatoriqn, ifacename) != IQN_FOUND) {
|
|
VIR_DEBUG("Failed to find interface '%s' with IQN '%s' "
|
|
"after attempting to create it",
|
|
&temp_ifacename[0], initiatoriqn);
|
|
goto cleanup;
|
|
} else {
|
|
VIR_DEBUG("Interface '%s' with IQN '%s' was created successfully",
|
|
*ifacename, initiatoriqn);
|
|
}
|
|
|
|
ret = 0;
|
|
|
|
cleanup:
|
|
virCommandFree(cmd);
|
|
VIR_FREE(temp_ifacename);
|
|
if (ret != 0)
|
|
VIR_FREE(*ifacename);
|
|
return ret;
|
|
}
|
|
|
|
|
|
static int
|
|
virISCSIConnection(const char *portal,
|
|
const char *initiatoriqn,
|
|
const char *target,
|
|
const char **extraargv)
|
|
{
|
|
int ret = -1;
|
|
const char *const baseargv[] = {
|
|
ISCSIADM,
|
|
"--mode", "node",
|
|
"--portal", portal,
|
|
"--targetname", target,
|
|
NULL
|
|
};
|
|
virCommandPtr cmd;
|
|
char *ifacename = NULL;
|
|
|
|
cmd = virCommandNewArgs(baseargv);
|
|
virCommandAddArgSet(cmd, extraargv);
|
|
|
|
if (initiatoriqn) {
|
|
switch (virStorageBackendIQNFound(initiatoriqn, &ifacename)) {
|
|
case IQN_FOUND:
|
|
VIR_DEBUG("ifacename: '%s'", ifacename);
|
|
break;
|
|
case IQN_MISSING:
|
|
if (virStorageBackendCreateIfaceIQN(initiatoriqn, &ifacename) != 0)
|
|
goto cleanup;
|
|
/*
|
|
* iscsiadm doesn't let you send commands to the Interface IQN,
|
|
* unless you've first issued a 'sendtargets' command to the
|
|
* portal. Without the sendtargets all that is received is a
|
|
* "iscsiadm: No records found"
|
|
*/
|
|
if (virISCSIScanTargets(portal, NULL, NULL) < 0)
|
|
goto cleanup;
|
|
|
|
break;
|
|
case IQN_ERROR:
|
|
default:
|
|
goto cleanup;
|
|
}
|
|
virCommandAddArgList(cmd, "--interface", ifacename, NULL);
|
|
}
|
|
|
|
if (virCommandRun(cmd, NULL) < 0)
|
|
goto cleanup;
|
|
|
|
ret = 0;
|
|
|
|
cleanup:
|
|
virCommandFree(cmd);
|
|
VIR_FREE(ifacename);
|
|
|
|
return ret;
|
|
}
|
|
|
|
|
|
int
|
|
virISCSIConnectionLogin(const char *portal,
|
|
const char *initiatoriqn,
|
|
const char *target)
|
|
{
|
|
const char *extraargv[] = { "--login", NULL };
|
|
return virISCSIConnection(portal, initiatoriqn, target, extraargv);
|
|
}
|
|
|
|
|
|
int
|
|
virISCSIConnectionLogout(const char *portal,
|
|
const char *initiatoriqn,
|
|
const char *target)
|
|
{
|
|
const char *extraargv[] = { "--logout", NULL };
|
|
return virISCSIConnection(portal, initiatoriqn, target, extraargv);
|
|
}
|
|
|
|
|
|
int
|
|
virISCSIRescanLUNs(const char *session)
|
|
{
|
|
virCommandPtr cmd = virCommandNewArgList(ISCSIADM,
|
|
"--mode", "session",
|
|
"-r", session,
|
|
"-R",
|
|
NULL);
|
|
int ret = virCommandRun(cmd, NULL);
|
|
virCommandFree(cmd);
|
|
return ret;
|
|
}
|
|
|
|
|
|
struct virISCSITargetList {
|
|
size_t ntargets;
|
|
char **targets;
|
|
};
|
|
|
|
|
|
static int
|
|
virISCSIGetTargets(char **const groups,
|
|
void *data)
|
|
{
|
|
struct virISCSITargetList *list = data;
|
|
char *target;
|
|
|
|
if (VIR_STRDUP(target, groups[1]) < 0)
|
|
return -1;
|
|
|
|
if (VIR_APPEND_ELEMENT(list->targets, list->ntargets, target) < 0) {
|
|
VIR_FREE(target);
|
|
return -1;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
|
|
int
|
|
virISCSIScanTargets(const char *portal,
|
|
size_t *ntargetsret,
|
|
char ***targetsret)
|
|
{
|
|
/**
|
|
*
|
|
* The output of sendtargets is very simple, just two columns,
|
|
* portal then target name
|
|
*
|
|
* 192.168.122.185:3260,1 iqn.2004-04.com:fedora14:iscsi.demo0.bf6d84
|
|
* 192.168.122.185:3260,1 iqn.2004-04.com:fedora14:iscsi.demo1.bf6d84
|
|
* 192.168.122.185:3260,1 iqn.2004-04.com:fedora14:iscsi.demo2.bf6d84
|
|
* 192.168.122.185:3260,1 iqn.2004-04.com:fedora14:iscsi.demo3.bf6d84
|
|
*/
|
|
const char *regexes[] = {
|
|
"^\\s*(\\S+)\\s+(\\S+)\\s*$"
|
|
};
|
|
int vars[] = { 2 };
|
|
struct virISCSITargetList list;
|
|
size_t i;
|
|
int ret = -1;
|
|
virCommandPtr cmd = virCommandNewArgList(ISCSIADM,
|
|
"--mode", "discovery",
|
|
"--type", "sendtargets",
|
|
"--portal", portal,
|
|
"--op", "nonpersistent",
|
|
NULL);
|
|
|
|
memset(&list, 0, sizeof(list));
|
|
|
|
if (virCommandRunRegex(cmd,
|
|
1,
|
|
regexes,
|
|
vars,
|
|
virISCSIGetTargets,
|
|
&list, NULL, NULL) < 0)
|
|
goto cleanup;
|
|
|
|
if (ntargetsret && targetsret) {
|
|
*ntargetsret = list.ntargets;
|
|
*targetsret = list.targets;
|
|
} else {
|
|
for (i = 0; i < list.ntargets; i++)
|
|
VIR_FREE(list.targets[i]);
|
|
VIR_FREE(list.targets);
|
|
}
|
|
|
|
ret = 0;
|
|
cleanup:
|
|
virCommandFree(cmd);
|
|
return ret;
|
|
}
|
|
|
|
/*
|
|
* virISCSINodeNew:
|
|
* @portal: address for iSCSI target
|
|
* @target: IQN and specific LUN target
|
|
*
|
|
* Usage of nonpersistent discovery in virISCSIScanTargets is useful primarily
|
|
* only when the target IQN is not known; however, since we have the target IQN
|
|
* usage of the "--op new" can be done. This avoids problems if "--op delete"
|
|
* had been used wiping out the static nodes determined by the scanning of
|
|
* all targets.
|
|
*
|
|
* NB: If an iSCSI node record is already created for this portal and
|
|
* target, subsequent "--op new" commands do not return an error.
|
|
*
|
|
* Returns 0 on success, -1 w/ error message on error
|
|
*/
|
|
int
|
|
virISCSINodeNew(const char *portal,
|
|
const char *target)
|
|
{
|
|
virCommandPtr cmd = NULL;
|
|
int status;
|
|
int ret = -1;
|
|
|
|
cmd = virCommandNewArgList(ISCSIADM,
|
|
"--mode", "node",
|
|
"--portal", portal,
|
|
"--targetname", target,
|
|
"--op", "new",
|
|
NULL);
|
|
|
|
if (virCommandRun(cmd, &status) < 0) {
|
|
virReportError(VIR_ERR_INTERNAL_ERROR,
|
|
_("Failed new node mode for target '%s'"),
|
|
target);
|
|
goto cleanup;
|
|
}
|
|
|
|
if (status != 0) {
|
|
virReportError(VIR_ERR_INTERNAL_ERROR,
|
|
_("%s failed new mode for target '%s' with status '%d'"),
|
|
ISCSIADM, target, status);
|
|
goto cleanup;
|
|
}
|
|
|
|
ret = 0;
|
|
cleanup:
|
|
virCommandFree(cmd);
|
|
return ret;
|
|
}
|
|
|
|
|
|
int
|
|
virISCSINodeUpdate(const char *portal,
|
|
const char *target,
|
|
const char *name,
|
|
const char *value)
|
|
{
|
|
virCommandPtr cmd = NULL;
|
|
int status;
|
|
int ret = -1;
|
|
|
|
cmd = virCommandNewArgList(ISCSIADM,
|
|
"--mode", "node",
|
|
"--portal", portal,
|
|
"--target", target,
|
|
"--op", "update",
|
|
"--name", name,
|
|
"--value", value,
|
|
NULL);
|
|
|
|
/* Ignore non-zero status. */
|
|
if (virCommandRun(cmd, &status) < 0) {
|
|
virReportError(VIR_ERR_INTERNAL_ERROR,
|
|
_("Failed to update '%s' of node mode for target '%s'"),
|
|
name, target);
|
|
goto cleanup;
|
|
}
|
|
|
|
ret = 0;
|
|
cleanup:
|
|
virCommandFree(cmd);
|
|
return ret;
|
|
}
|