mirror of
https://gitlab.com/libvirt/libvirt.git
synced 2025-01-08 14:05:19 +00:00
b15093d867
Signed-off-by: Ján Tomko <jtomko@redhat.com> Reviewed-by: Erik Skultety <eskultet@redhat.com>
452 lines
11 KiB
C
452 lines
11 KiB
C
/*
|
|
* virscsi.c: helper APIs for managing host SCSI devices
|
|
*
|
|
* Copyright (C) 2013-2014 Red Hat, Inc.
|
|
* Copyright (C) 2013 Fujitsu, Inc.
|
|
*
|
|
* This library is free software; you can redistribute it and/or
|
|
* modify it under the terms of the GNU Lesser General Public
|
|
* License as published by the Free Software Foundation; either
|
|
* version 2.1 of the License, or (at your option) any later version.
|
|
*
|
|
* This library is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
* Lesser General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU Lesser General Public
|
|
* License along with this library. If not, see
|
|
* <http://www.gnu.org/licenses/>.
|
|
*/
|
|
|
|
#include <config.h>
|
|
|
|
#include <dirent.h>
|
|
#include <fcntl.h>
|
|
#include <inttypes.h>
|
|
#include <sys/types.h>
|
|
#include <sys/stat.h>
|
|
#include <unistd.h>
|
|
|
|
#include "virlog.h"
|
|
#include "virscsi.h"
|
|
#include "virfile.h"
|
|
#include "virstring.h"
|
|
#include "virerror.h"
|
|
#include "viralloc.h"
|
|
|
|
#define SYSFS_SCSI_DEVICES "/sys/bus/scsi/devices"
|
|
|
|
/* For virReportOOMError() and virReportSystemError() */
|
|
#define VIR_FROM_THIS VIR_FROM_NONE
|
|
|
|
VIR_LOG_INIT("util.scsi");
|
|
|
|
struct _virUsedByInfo {
|
|
char *drvname; /* which driver */
|
|
char *domname; /* which domain */
|
|
};
|
|
typedef struct _virUsedByInfo virUsedByInfo;
|
|
typedef virUsedByInfo *virUsedByInfoPtr;
|
|
|
|
struct _virSCSIDevice {
|
|
unsigned int adapter;
|
|
unsigned int bus;
|
|
unsigned int target;
|
|
unsigned long long unit;
|
|
|
|
char *name; /* adapter:bus:target:unit */
|
|
char *id; /* model:vendor */
|
|
char *sg_path; /* e.g. /dev/sg2 */
|
|
virUsedByInfoPtr *used_by; /* driver:domain(s) using this dev */
|
|
size_t n_used_by; /* how many domains are using this dev */
|
|
|
|
bool readonly;
|
|
bool shareable;
|
|
};
|
|
|
|
struct _virSCSIDeviceList {
|
|
virObjectLockable parent;
|
|
size_t count;
|
|
virSCSIDevicePtr *devs;
|
|
};
|
|
|
|
static virClassPtr virSCSIDeviceListClass;
|
|
|
|
static void virSCSIDeviceListDispose(void *obj);
|
|
|
|
static int
|
|
virSCSIOnceInit(void)
|
|
{
|
|
if (!VIR_CLASS_NEW(virSCSIDeviceList, virClassForObjectLockable()))
|
|
return -1;
|
|
|
|
return 0;
|
|
}
|
|
|
|
VIR_ONCE_GLOBAL_INIT(virSCSI);
|
|
|
|
static int
|
|
virSCSIDeviceGetAdapterId(const char *adapter,
|
|
unsigned int *adapter_id)
|
|
{
|
|
if (STRPREFIX(adapter, "scsi_host") &&
|
|
virStrToLong_ui(adapter + strlen("scsi_host"),
|
|
NULL, 0, adapter_id) == 0)
|
|
return 0;
|
|
virReportError(VIR_ERR_INTERNAL_ERROR,
|
|
_("Cannot parse adapter '%s'"), adapter);
|
|
return -1;
|
|
}
|
|
|
|
char *
|
|
virSCSIDeviceGetSgName(const char *sysfs_prefix,
|
|
const char *adapter,
|
|
unsigned int bus,
|
|
unsigned int target,
|
|
unsigned long long unit)
|
|
{
|
|
DIR *dir = NULL;
|
|
struct dirent *entry;
|
|
g_autofree char *path = NULL;
|
|
char *sg = NULL;
|
|
unsigned int adapter_id;
|
|
const char *prefix = sysfs_prefix ? sysfs_prefix : SYSFS_SCSI_DEVICES;
|
|
|
|
if (virSCSIDeviceGetAdapterId(adapter, &adapter_id) < 0)
|
|
return NULL;
|
|
|
|
path = g_strdup_printf("%s/%d:%u:%u:%llu/scsi_generic", prefix, adapter_id,
|
|
bus, target, unit);
|
|
|
|
if (virDirOpen(&dir, path) < 0)
|
|
goto cleanup;
|
|
|
|
while (virDirRead(dir, &entry, path) > 0) {
|
|
/* Assume a single directory entry */
|
|
sg = g_strdup(entry->d_name);
|
|
break;
|
|
}
|
|
|
|
cleanup:
|
|
VIR_DIR_CLOSE(dir);
|
|
return sg;
|
|
}
|
|
|
|
/* Returns device name (e.g. "sdc") on success, or NULL
|
|
* on failure.
|
|
*/
|
|
char *
|
|
virSCSIDeviceGetDevName(const char *sysfs_prefix,
|
|
const char *adapter,
|
|
unsigned int bus,
|
|
unsigned int target,
|
|
unsigned long long unit)
|
|
{
|
|
DIR *dir = NULL;
|
|
struct dirent *entry;
|
|
g_autofree char *path = NULL;
|
|
char *name = NULL;
|
|
unsigned int adapter_id;
|
|
const char *prefix = sysfs_prefix ? sysfs_prefix : SYSFS_SCSI_DEVICES;
|
|
|
|
if (virSCSIDeviceGetAdapterId(adapter, &adapter_id) < 0)
|
|
return NULL;
|
|
|
|
path = g_strdup_printf("%s/%d:%u:%u:%llu/block", prefix, adapter_id, bus,
|
|
target, unit);
|
|
|
|
if (virDirOpen(&dir, path) < 0)
|
|
goto cleanup;
|
|
|
|
while (virDirRead(dir, &entry, path) > 0) {
|
|
name = g_strdup(entry->d_name);
|
|
break;
|
|
}
|
|
|
|
cleanup:
|
|
VIR_DIR_CLOSE(dir);
|
|
return name;
|
|
}
|
|
|
|
virSCSIDevicePtr
|
|
virSCSIDeviceNew(const char *sysfs_prefix,
|
|
const char *adapter,
|
|
unsigned int bus,
|
|
unsigned int target,
|
|
unsigned long long unit,
|
|
bool readonly,
|
|
bool shareable)
|
|
{
|
|
g_autoptr(virSCSIDevice) dev = NULL;
|
|
g_autofree char *sg = NULL;
|
|
g_autofree char *vendor_path = NULL;
|
|
g_autofree char *model_path = NULL;
|
|
g_autofree char *vendor = NULL;
|
|
g_autofree char *model = NULL;
|
|
const char *prefix = sysfs_prefix ? sysfs_prefix : SYSFS_SCSI_DEVICES;
|
|
|
|
dev = g_new0(virSCSIDevice, 1);
|
|
|
|
dev->bus = bus;
|
|
dev->target = target;
|
|
dev->unit = unit;
|
|
dev->readonly = readonly;
|
|
dev->shareable = shareable;
|
|
|
|
if (!(sg = virSCSIDeviceGetSgName(prefix, adapter, bus, target, unit)))
|
|
return NULL;
|
|
|
|
if (virSCSIDeviceGetAdapterId(adapter, &dev->adapter) < 0)
|
|
return NULL;
|
|
|
|
dev->name = g_strdup_printf("%d:%u:%u:%llu", dev->adapter,
|
|
dev->bus, dev->target, dev->unit);
|
|
dev->sg_path = g_strdup_printf("%s/%s",
|
|
sysfs_prefix ? sysfs_prefix : "/dev", sg);
|
|
|
|
if (!virFileExists(dev->sg_path)) {
|
|
virReportSystemError(errno,
|
|
_("SCSI device '%s': could not access %s"),
|
|
dev->name, dev->sg_path);
|
|
return NULL;
|
|
}
|
|
|
|
vendor_path = g_strdup_printf("%s/%s/vendor", prefix, dev->name);
|
|
model_path = g_strdup_printf("%s/%s/model", prefix, dev->name);
|
|
|
|
if (virFileReadAll(vendor_path, 1024, &vendor) < 0)
|
|
return NULL;
|
|
|
|
if (virFileReadAll(model_path, 1024, &model) < 0)
|
|
return NULL;
|
|
|
|
virTrimSpaces(vendor, NULL);
|
|
virTrimSpaces(model, NULL);
|
|
|
|
dev->id = g_strdup_printf("%s:%s", vendor, model);
|
|
|
|
return g_steal_pointer(&dev);
|
|
}
|
|
|
|
static void
|
|
virSCSIDeviceUsedByInfoFree(virUsedByInfoPtr used_by)
|
|
{
|
|
VIR_FREE(used_by->drvname);
|
|
VIR_FREE(used_by->domname);
|
|
VIR_FREE(used_by);
|
|
}
|
|
G_DEFINE_AUTOPTR_CLEANUP_FUNC(virUsedByInfo, virSCSIDeviceUsedByInfoFree);
|
|
|
|
void
|
|
virSCSIDeviceFree(virSCSIDevicePtr dev)
|
|
{
|
|
size_t i;
|
|
|
|
if (!dev)
|
|
return;
|
|
|
|
VIR_FREE(dev->id);
|
|
VIR_FREE(dev->name);
|
|
VIR_FREE(dev->sg_path);
|
|
for (i = 0; i < dev->n_used_by; i++)
|
|
virSCSIDeviceUsedByInfoFree(dev->used_by[i]);
|
|
VIR_FREE(dev->used_by);
|
|
VIR_FREE(dev);
|
|
}
|
|
|
|
int
|
|
virSCSIDeviceSetUsedBy(virSCSIDevicePtr dev,
|
|
const char *drvname,
|
|
const char *domname)
|
|
{
|
|
g_autoptr(virUsedByInfo) copy = NULL;
|
|
|
|
copy = g_new0(virUsedByInfo, 1);
|
|
copy->drvname = g_strdup(drvname);
|
|
copy->domname = g_strdup(domname);
|
|
|
|
if (VIR_APPEND_ELEMENT(dev->used_by, dev->n_used_by, copy) < 0)
|
|
return -1;
|
|
|
|
return 0;
|
|
}
|
|
|
|
bool
|
|
virSCSIDeviceIsAvailable(virSCSIDevicePtr dev)
|
|
{
|
|
return dev->n_used_by == 0;
|
|
}
|
|
|
|
const char *
|
|
virSCSIDeviceGetName(virSCSIDevicePtr dev)
|
|
{
|
|
return dev->name;
|
|
}
|
|
|
|
const char *
|
|
virSCSIDeviceGetPath(virSCSIDevicePtr dev)
|
|
{
|
|
return dev->sg_path;
|
|
}
|
|
|
|
unsigned int
|
|
virSCSIDeviceGetAdapter(virSCSIDevicePtr dev)
|
|
{
|
|
return dev->adapter;
|
|
}
|
|
|
|
unsigned int
|
|
virSCSIDeviceGetBus(virSCSIDevicePtr dev)
|
|
{
|
|
return dev->bus;
|
|
}
|
|
|
|
unsigned int
|
|
virSCSIDeviceGetTarget(virSCSIDevicePtr dev)
|
|
{
|
|
return dev->target;
|
|
}
|
|
|
|
unsigned long long
|
|
virSCSIDeviceGetUnit(virSCSIDevicePtr dev)
|
|
{
|
|
return dev->unit;
|
|
}
|
|
|
|
bool
|
|
virSCSIDeviceGetReadonly(virSCSIDevicePtr dev)
|
|
{
|
|
return dev->readonly;
|
|
}
|
|
|
|
bool
|
|
virSCSIDeviceGetShareable(virSCSIDevicePtr dev)
|
|
{
|
|
return dev->shareable;
|
|
}
|
|
|
|
int
|
|
virSCSIDeviceFileIterate(virSCSIDevicePtr dev,
|
|
virSCSIDeviceFileActor actor,
|
|
void *opaque)
|
|
{
|
|
return (actor)(dev, dev->sg_path, opaque);
|
|
}
|
|
|
|
virSCSIDeviceListPtr
|
|
virSCSIDeviceListNew(void)
|
|
{
|
|
virSCSIDeviceListPtr list;
|
|
|
|
if (virSCSIInitialize() < 0)
|
|
return NULL;
|
|
|
|
if (!(list = virObjectLockableNew(virSCSIDeviceListClass)))
|
|
return NULL;
|
|
|
|
return list;
|
|
}
|
|
|
|
static void
|
|
virSCSIDeviceListDispose(void *obj)
|
|
{
|
|
virSCSIDeviceListPtr list = obj;
|
|
size_t i;
|
|
|
|
for (i = 0; i < list->count; i++)
|
|
virSCSIDeviceFree(list->devs[i]);
|
|
|
|
VIR_FREE(list->devs);
|
|
}
|
|
|
|
int
|
|
virSCSIDeviceListAdd(virSCSIDeviceListPtr list,
|
|
virSCSIDevicePtr dev)
|
|
{
|
|
if (virSCSIDeviceListFind(list, dev)) {
|
|
virReportError(VIR_ERR_INTERNAL_ERROR,
|
|
_("Device %s already exists"),
|
|
dev->name);
|
|
return -1;
|
|
}
|
|
|
|
return VIR_APPEND_ELEMENT(list->devs, list->count, dev);
|
|
}
|
|
|
|
virSCSIDevicePtr
|
|
virSCSIDeviceListGet(virSCSIDeviceListPtr list, int idx)
|
|
{
|
|
if (idx >= list->count || idx < 0)
|
|
return NULL;
|
|
|
|
return list->devs[idx];
|
|
}
|
|
|
|
size_t
|
|
virSCSIDeviceListCount(virSCSIDeviceListPtr list)
|
|
{
|
|
return list->count;
|
|
}
|
|
|
|
virSCSIDevicePtr
|
|
virSCSIDeviceListSteal(virSCSIDeviceListPtr list,
|
|
virSCSIDevicePtr dev)
|
|
{
|
|
virSCSIDevicePtr ret = NULL;
|
|
size_t i;
|
|
|
|
for (i = 0; i < list->count; i++) {
|
|
if (list->devs[i]->adapter == dev->adapter &&
|
|
list->devs[i]->bus == dev->bus &&
|
|
list->devs[i]->target == dev->target &&
|
|
list->devs[i]->unit == dev->unit) {
|
|
ret = list->devs[i];
|
|
VIR_DELETE_ELEMENT(list->devs, i, list->count);
|
|
break;
|
|
}
|
|
}
|
|
|
|
return ret;
|
|
}
|
|
|
|
void
|
|
virSCSIDeviceListDel(virSCSIDeviceListPtr list,
|
|
virSCSIDevicePtr dev,
|
|
const char *drvname,
|
|
const char *domname)
|
|
{
|
|
size_t i;
|
|
|
|
for (i = 0; i < dev->n_used_by; i++) {
|
|
if (STREQ_NULLABLE(dev->used_by[i]->drvname, drvname) &&
|
|
STREQ_NULLABLE(dev->used_by[i]->domname, domname)) {
|
|
if (dev->n_used_by > 1) {
|
|
virSCSIDeviceUsedByInfoFree(dev->used_by[i]);
|
|
VIR_DELETE_ELEMENT(dev->used_by, i, dev->n_used_by);
|
|
} else {
|
|
g_autoptr(virSCSIDevice) tmp = NULL;
|
|
tmp = virSCSIDeviceListSteal(list, dev);
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
virSCSIDevicePtr
|
|
virSCSIDeviceListFind(virSCSIDeviceListPtr list,
|
|
virSCSIDevicePtr dev)
|
|
{
|
|
size_t i;
|
|
|
|
for (i = 0; i < list->count; i++) {
|
|
if (list->devs[i]->adapter == dev->adapter &&
|
|
list->devs[i]->bus == dev->bus &&
|
|
list->devs[i]->target == dev->target &&
|
|
list->devs[i]->unit == dev->unit)
|
|
return list->devs[i];
|
|
}
|
|
|
|
return NULL;
|
|
}
|