[libvirt] [RFC PATCH v2 REBASE 01/18] util: Introduce new module virmdev

Pavel Hrdina phrdina at redhat.com
Tue Feb 28 08:22:26 UTC 2017


On Mon, Feb 20, 2017 at 03:28:14PM +0100, Erik Skultety wrote:
> Beside creation, disposal, getter, and setter methods the module exports
> methods to work with lists of mediated devices.
> 
> Signed-off-by: Erik Skultety <eskultet at redhat.com>
> ---
>  po/POTFILES.in           |   1 +
>  src/Makefile.am          |   1 +
>  src/libvirt_private.syms |  18 +++
>  src/util/virmdev.c       | 358 +++++++++++++++++++++++++++++++++++++++++++++++
>  src/util/virmdev.h       |  93 ++++++++++++
>  5 files changed, 471 insertions(+)
>  create mode 100644 src/util/virmdev.c
>  create mode 100644 src/util/virmdev.h
> 
> diff --git a/po/POTFILES.in b/po/POTFILES.in
> index 9f66697..c857211 100644
> --- a/po/POTFILES.in
> +++ b/po/POTFILES.in
> @@ -218,6 +218,7 @@ src/util/virlease.c
>  src/util/virlockspace.c
>  src/util/virlog.c
>  src/util/virmacmap.c
> +src/util/virmdev.c
>  src/util/virnetdev.c
>  src/util/virnetdevbandwidth.c
>  src/util/virnetdevbridge.c
> diff --git a/src/Makefile.am b/src/Makefile.am
> index 46ca272..7bc2d3b 100644
> --- a/src/Makefile.am
> +++ b/src/Makefile.am
> @@ -188,6 +188,7 @@ UTIL_SOURCES =							\
>  		util/virvhba.c util/virvhba.h			\
>  		util/virxdrdefs.h                               \
>  		util/virxml.c util/virxml.h			\
> +		util/virmdev.c util/virmdev.h			\
>  		$(NULL)
>  
>  EXTRA_DIST += $(srcdir)/util/keymaps.csv $(srcdir)/util/virkeycode-mapgen.py
> diff --git a/src/libvirt_private.syms b/src/libvirt_private.syms
> index e6ccd69..8af65a1 100644
> --- a/src/libvirt_private.syms
> +++ b/src/libvirt_private.syms
> @@ -1959,6 +1959,24 @@ virMacMapNew;
>  virMacMapRemove;
>  virMacMapWriteFile;
>  
> +# util/virmdev.h
> +virMediatedDeviceFree;
> +virMediatedDeviceGetDeviceAPI;
> +virMediatedDeviceGetIOMMUGroupDev;
> +virMediatedDeviceGetPath;
> +virMediatedDeviceGetUsedBy;
> +virMediatedDeviceListAdd;
> +virMediatedDeviceListCount;
> +virMediatedDeviceListDel;
> +virMediatedDeviceListFind;
> +virMediatedDeviceListGet;
> +virMediatedDeviceListNew;
> +virMediatedDeviceListSteal;
> +virMediatedDeviceListStealIndex;
> +virMediatedDeviceNew;
> +virMediatedDeviceSetUsedBy;
> +
> +
>  
>  # util/virnetdev.h
>  virNetDevAddMulti;
> diff --git a/src/util/virmdev.c b/src/util/virmdev.c
> new file mode 100644
> index 0000000..9c80647
> --- /dev/null
> +++ b/src/util/virmdev.c
> @@ -0,0 +1,358 @@
> +/*
> + * virmdev.c: helper APIs for managing host MDEV devices
> + *
> + * Copyright (C) 2017-2018 Red Hat, Inc.

s/2017-2018/2017/

> + *
> + * 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 <fcntl.h>
> +#include <inttypes.h>
> +#include <limits.h>
> +#include <stdio.h>
> +#include <string.h>
> +#include <sys/types.h>
> +#include <sys/stat.h>
> +#include <unistd.h>
> +#include <stdlib.h>
> +
> +#include "virmdev.h"
> +#include "dirname.h"
> +#include "virlog.h"
> +#include "viralloc.h"
> +#include "vircommand.h"
> +#include "virerror.h"
> +#include "virfile.h"
> +#include "virkmod.h"
> +#include "virstring.h"
> +#include "virutil.h"
> +#include "viruuid.h"
> +#include "virhostdev.h"
> +
> +VIR_LOG_INIT("util.mdev");
> +
> +struct _virMediatedDevice {
> +    char *path;                             /* sysfs path */
> +
> +    char *used_by_drvname;
> +    char *used_by_domname;
> +};
> +
> +struct _virMediatedDeviceList {
> +    virObjectLockable parent;
> +
> +    size_t count;
> +    virMediatedDevicePtr *devs;
> +};
> +
> +
> +/* For virReportOOMError()  and virReportSystemError() */
> +#define VIR_FROM_THIS VIR_FROM_NONE

We usually place this define right after all includes and I think that you
can drop the comment.

> +
> +static virClassPtr virMediatedDeviceListClass;
> +
> +static void virMediatedDeviceListDispose(void *obj);
> +
> +static int virMediatedOnceInit(void)
> +{
> +    if (!(virMediatedDeviceListClass = virClassNew(virClassForObjectLockable(),
> +                                                   "virMediatedDeviceList",
> +                                                   sizeof(virMediatedDeviceList),
> +                                                   virMediatedDeviceListDispose)))
> +        return -1;
> +
> +    return 0;
> +}
> +
> +VIR_ONCE_GLOBAL_INIT(virMediated)
> +
> +#ifdef __linux__
> +# define MDEV_SYSFS_DEVICES "/sys/bus/mdev/devices/"
> +
> +virMediatedDevicePtr
> +virMediatedDeviceNew(const char *uuidstr)
> +{
> +    virMediatedDevicePtr dev = NULL, ret = NULL;
> +
> +    if (VIR_ALLOC(dev) < 0)
> +        return NULL;
> +
> +    if (virAsprintf(&dev->path, MDEV_SYSFS_DEVICES "%s", uuidstr) < 0)
> +       goto cleanup;
> +
> +    ret = dev;
> +    dev = NULL;
> +
> + cleanup:
> +    virMediatedDeviceFree(dev);
> +    return ret;
> +}
> +
> +#else
> +
> +virMediatedDevicePtr
> +virMediatedDeviceNew(virPCIDeviceAddressPtr pciaddr ATTRIBUTE_UNUSED,
> +                     const char *uuidstr ATTRIBUTE_UNUSED)
> +{
> +    virRerportError(VIR_ERR_INTERNAL_ERROR, "%s",
> +                    _("not supported on non-linux platforms"));
> +    return NULL;
> +}
> +
> +#endif /* __linux__ */
> +
> +void
> +virMediatedDeviceFree(virMediatedDevicePtr dev)
> +{
> +    if (!dev)
> +        return;
> +    VIR_FREE(dev->path);
> +    VIR_FREE(dev->used_by_drvname);
> +    VIR_FREE(dev->used_by_domname);
> +    VIR_FREE(dev);
> +}
> +
> +
> +const char *
> +virMediatedDeviceGetPath(virMediatedDevicePtr dev)
> +{
> +    return dev->path;
> +}
> +
> +
> +/* Returns an absolute canonicalized path to the device used to control the
> + * mediated device's IOMMU group (e.g. "/dev/vfio/15")
> + */
> +char *
> +virMediatedDeviceGetIOMMUGroupDev(virMediatedDevicePtr dev)
> +{
> +    char *resultpath = NULL;
> +    char *iommu_linkpath = NULL;
> +    char *vfio_path = NULL;
> +
> +    if (virAsprintf(&iommu_linkpath, "%s/iommu_group", dev->path) < 0)
> +        return NULL;
> +
> +    if (virFileIsLink(iommu_linkpath) != 1) {
> +        virReportError(VIR_ERR_INTERNAL_ERROR,
> +                       _("IOMMU group file %s is not a symlink"),
> +                       iommu_linkpath);
> +        goto cleanup;
> +    }
> +
> +    if (virFileResolveLink(iommu_linkpath, &resultpath) < 0) {
> +        virReportError(VIR_ERR_INTERNAL_ERROR,
> +                       _("Unable to resolve IOMMU group symlink %s"),
> +                       iommu_linkpath);
> +        goto cleanup;
> +    }
> +
> +    if (virAsprintf(&vfio_path, "/dev/vfio/%s", last_component(resultpath)) < 0)
> +        goto cleanup;
> +
> + cleanup:
> +    VIR_FREE(resultpath);
> +    VIR_FREE(iommu_linkpath);
> +    return vfio_path;
> +}
> +
> +
> +void
> +virMediatedDeviceGetUsedBy(virMediatedDevicePtr dev,
> +                           const char **drvname, const char **domname)
> +{
> +    *drvname = dev->used_by_drvname;
> +    *domname = dev->used_by_domname;
> +}
> +
> +
> +int
> +virMediatedDeviceSetUsedBy(virMediatedDevicePtr dev,
> +                           const char *drvname,
> +                           const char *domname)
> +{
> +    VIR_FREE(dev->used_by_drvname);
> +    VIR_FREE(dev->used_by_domname);
> +    if (VIR_STRDUP(dev->used_by_drvname, drvname) < 0)
> +        return -1;
> +    if (VIR_STRDUP(dev->used_by_domname, domname) < 0)
> +        return -1;
> +
> +    return 0;
> +}
> +
> +
> +virMediatedDeviceListPtr
> +virMediatedDeviceListNew(void)
> +{
> +    virMediatedDeviceListPtr list;
> +
> +    if (virMediatedInitialize() < 0)
> +        return NULL;
> +
> +    if (!(list = virObjectLockableNew(virMediatedDeviceListClass)))
> +        return NULL;
> +
> +    return list;
> +}
> +
> +
> +static void
> +virMediatedDeviceListDispose(void *obj)
> +{
> +    virMediatedDeviceListPtr list = obj;
> +    size_t i;
> +
> +    for (i = 0; i < list->count; i++) {
> +        virMediatedDeviceFree(list->devs[i]);
> +        list->devs[i] = NULL;
> +    }
> +
> +    list->count = 0;
> +    VIR_FREE(list->devs);
> +}
> +
> +
> +int
> +virMediatedDeviceListAdd(virMediatedDeviceListPtr list,
> +                         virMediatedDevicePtr dev)
> +{
> +    if (virMediatedDeviceListFind(list, dev)) {
> +        virReportError(VIR_ERR_INTERNAL_ERROR,
> +                       _("Device %s is already in use"), dev->path);
> +        return -1;
> +    }
> +    return VIR_APPEND_ELEMENT(list->devs, list->count, dev);
> +}
> +
> +
> +virMediatedDevicePtr
> +virMediatedDeviceListGet(virMediatedDeviceListPtr list,
> +                         ssize_t idx)
> +{
> +    if (idx < 0 || idx >= list->count)
> +        return NULL;
> +
> +    return list->devs[idx];
> +}
> +
> +
> +size_t
> +virMediatedDeviceListCount(virMediatedDeviceListPtr list)
> +{
> +    return list->count;
> +}
> +
> +
> +virMediatedDevicePtr
> +virMediatedDeviceListStealIndex(virMediatedDeviceListPtr list,
> +                                ssize_t idx)
> +{
> +    virMediatedDevicePtr ret;
> +
> +    if (idx < 0 || idx >= list->count)
> +        return NULL;
> +
> +    ret = list->devs[idx];
> +    VIR_DELETE_ELEMENT(list->devs, idx, list->count);
> +    return ret;
> +}
> +
> +
> +virMediatedDevicePtr
> +virMediatedDeviceListSteal(virMediatedDeviceListPtr list,
> +                           virMediatedDevicePtr dev)
> +{
> +    int idx = virMediatedDeviceListFindIndex(list, dev);
> +
> +    return virMediatedDeviceListStealIndex(list, idx);
> +}
> +
> +
> +void
> +virMediatedDeviceListDel(virMediatedDeviceListPtr list,
> +                         virMediatedDevicePtr dev)
> +{
> +    virMediatedDevicePtr ret = virMediatedDeviceListSteal(list, dev);
> +    virMediatedDeviceFree(ret);
> +}
> +
> +
> +int
> +virMediatedDeviceListFindIndex(virMediatedDeviceListPtr list,
> +                               virMediatedDevicePtr dev)
> +{
> +    size_t i;
> +
> +    for (i = 0; i < list->count; i++) {
> +        virMediatedDevicePtr other = list->devs[i];
> +        if (STREQ(other->path, dev->path))
> +            return i;
> +    }
> +    return -1;
> +}
> +
> +
> +virMediatedDevicePtr
> +virMediatedDeviceListFind(virMediatedDeviceListPtr list,
> +                          virMediatedDevicePtr dev)
> +{
> +    int idx;
> +
> +    if ((idx = virMediatedDeviceListFindIndex(list, dev)) >= 0)
> +        return list->devs[idx];
> +    else
> +        return NULL;
> +}
> +
> +
> +int
> +virMediatedDeviceGetDeviceAPI(virMediatedDevicePtr dev,
> +                              char **device_api)
> +{
> +    int ret = -1;
> +    char *buf = NULL;
> +    char *tmp = NULL;
> +    char *sysfs_path = NULL;
> +
> +    if (virAsprintf(&sysfs_path, "%s/mdev_type/device_api", dev->path) < 0)
> +        goto cleanup;
> +
> +    /* TODO - make this a generic method to access sysfs files for various
> +     * kinds of devices
> +     */
> +    if (!virFileExists(sysfs_path)) {
> +        virReportError(VIR_ERR_OPERATION_INVALID, "%s",
> +                       _("mediated devices are not supported by this kernel"));
> +        goto cleanup;
> +    }
> +
> +    if (virFileReadAll(sysfs_path, 1024, &buf) < 0)
> +        goto cleanup;
> +
> +    if ((tmp = strchr(buf, '\n')))
> +        *tmp = '\0';
> +
> +    *device_api = buf;
> +    buf = NULL;
> +
> +    ret = 0;
> + cleanup:
> +    VIR_FREE(sysfs_path);
> +    VIR_FREE(buf);
> +    return ret;
> +}
> diff --git a/src/util/virmdev.h b/src/util/virmdev.h
> new file mode 100644
> index 0000000..78a7df5
> --- /dev/null
> +++ b/src/util/virmdev.h
> @@ -0,0 +1,93 @@
> +/*
> + * virmdev.h: helper APIs for managing host mediated devices
> + *
> + * Copyright (C) 2017-2018 Red Hat, Inc.

s/2017-2018/2017/

> + *
> + * This library is free software; you can redistribute it and/or
> + * modify it under the terms of the GNU Lesser General Public
> + * License as published by the Free Software Foundation; either
> + * version 2.1 of the License, or (at your option) any later version.
> + *
> + * This library is distributed in the hope that it will be useful,
> + * but WITHOUT ANY WARRANTY; without even the implied warranty of
> + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
> + * Lesser General Public License for more details.
> + *
> + * You should have received a copy of the GNU Lesser General Public
> + * License along with this library.  If not, see
> + * <http://www.gnu.org/licenses/>.
> + */
> +
> +#ifndef __VIR_MDEV_H__
> +# define __VIR_MDEV_H__
> +
> +# include "internal.h"
> +# include "virobject.h"
> +# include "virutil.h"
> +# include "virpci.h"
> +
> +typedef enum {
> +    VIR_MDEV_MODEL_TYPE_DEFAULT = 0,
> +    VIR_MDEV_MODEL_TYPE_VFIO_PCI,
> +
> +    VIR_MDEV_MODEL_TYPE_LAST
> +} virMediatedDeviceModelType;
> +
> +VIR_ENUM_DECL(virMediatedDeviceModel)
> +
> +
> +typedef struct _virMediatedDevice virMediatedDevice;
> +typedef virMediatedDevice *virMediatedDevicePtr;
> +typedef struct _virMediatedDeviceAddress virMediatedDeviceAddress;
> +typedef virMediatedDeviceAddress *virMediatedDeviceAddressPtr;
> +typedef struct _virMediatedDeviceList virMediatedDeviceList;
> +typedef virMediatedDeviceList *virMediatedDeviceListPtr;
> +
> +typedef int (*virMediatedDeviceCallback)(virMediatedDevicePtr dev,
> +                                         const char *path, void *opaque);
> +
> +virMediatedDevicePtr virMediatedDeviceNew(const char *uuidstr);

I think it's time to start using the same format as for .c files especially
when introducing new header files.

Pavel

> +
> +virMediatedDevicePtr virMediatedDeviceCopy(virMediatedDevicePtr dev);
> +
> +void virMediatedDeviceFree(virMediatedDevicePtr dev);
> +
> +const char *virMediatedDeviceGetPath(virMediatedDevicePtr dev);
> +
> +void virMediatedDeviceGetUsedBy(virMediatedDevicePtr dev,
> +                                const char **drvname, const char **domname);
> +
> +int virMediatedDeviceSetUsedBy(virMediatedDevicePtr dev,
> +                               const char *drvname,
> +                               const char *domname);
> +
> +char *virMediatedDeviceGetIOMMUGroupDev(virMediatedDevicePtr dev);
> +
> +int virMediatedDeviceGetDeviceAPI(virMediatedDevicePtr dev, char **device_api);
> +
> +virMediatedDeviceListPtr virMediatedDeviceListNew(void);
> +
> +int  virMediatedDeviceListAdd(virMediatedDeviceListPtr list,
> +                              virMediatedDevicePtr dev);
> +
> +virMediatedDevicePtr virMediatedDeviceListGet(virMediatedDeviceListPtr list,
> +                                              ssize_t idx);
> +
> +size_t virMediatedDeviceListCount(virMediatedDeviceListPtr list);
> +
> +virMediatedDevicePtr virMediatedDeviceListSteal(virMediatedDeviceListPtr list,
> +                                                virMediatedDevicePtr dev);
> +
> +virMediatedDevicePtr virMediatedDeviceListStealIndex(virMediatedDeviceListPtr list,
> +                                                     ssize_t idx);
> +
> +void virMediatedDeviceListDel(virMediatedDeviceListPtr list,
> +                              virMediatedDevicePtr dev);
> +
> +virMediatedDevicePtr virMediatedDeviceListFind(virMediatedDeviceListPtr list,
> +                                               virMediatedDevicePtr dev);
> +
> +int virMediatedDeviceListFindIndex(virMediatedDeviceListPtr list,
> +                                   virMediatedDevicePtr dev);
> +
> +#endif /* __VIR_MDEV_H__ */
> -- 
> 2.10.2
> 
> --
> libvir-list mailing list
> libvir-list at redhat.com
> https://www.redhat.com/mailman/listinfo/libvir-list
-------------- next part --------------
A non-text attachment was scrubbed...
Name: signature.asc
Type: application/pgp-signature
Size: 833 bytes
Desc: Digital signature
URL: <http://listman.redhat.com/archives/libvir-list/attachments/20170228/9ec23dba/attachment-0001.sig>


More information about the libvir-list mailing list