mirror of
https://gitlab.com/libvirt/libvirt.git
synced 2024-11-02 11:21:12 +00:00
2e56fb2bcc
Prior to this patch, there was an inconsistent mix between GNU and C99. For consistency, and potential portability to other compilers, stick with the C99 vararg macro syntax. * src/conf/cpu_conf.c (virCPUReportError): Use C99 rather than GNU vararg macro syntax. * src/conf/domain_conf.c (virDomainReportError): Likewise. * src/conf/domain_event.c (eventReportError): Likewise. * src/conf/interface_conf.c (virInterfaceReportError): Likewise. * src/conf/network_conf.c (virNetworkReportError): Likewise. * src/conf/node_device_conf.h (virNodeDeviceReportError): Likewise. * src/conf/secret_conf.h (virSecretReportError): Likewise. * src/conf/storage_conf.h (virStorageReportError): Likewise. * src/esx/esx_device_monitor.c (ESX_ERROR): Use C99 rather than GNU vararg macro syntax. * src/esx/esx_driver.c (ESX_ERROR): Likewise. * src/esx/esx_interface_driver.c (ESX_ERROR): Likewise. * src/esx/esx_network_driver.c (ESX_ERROR): Likewise. * src/esx/esx_secret_driver.c (ESX_ERROR): Likewise. * src/esx/esx_storage_driver.c (ESX_ERROR): Likewise. * src/esx/esx_util.c (ESX_ERROR): Likewise. * src/esx/esx_vi.c (ESX_VI_ERROR): Likewise. * src/esx/esx_vi_methods.c (ESX_VI_ERROR): Likewise. * src/esx/esx_vi_types.c (ESX_VI_ERROR): Likewise. * src/esx/esx_vmx.c (ESX_ERROR): Likewise. * src/util/hostusb.c (usbReportError): Use C99 rather than GNU vararg macro syntax. * src/util/json.c (virJSONError): Likewise. * src/util/macvtap.c (ReportError): Likewise. * src/util/pci.c (pciReportError): Likewise. * src/util/stats_linux.c (virStatsError): Likewise. * src/util/util.c (virUtilError): Likewise. * src/util/xml.c (virXMLError): Likewise. * src/xen/proxy_internal.c (virProxyError): Use C99 rather than GNU vararg macro syntax. * src/xen/sexpr.c (virSexprError): Likewise. * src/xen/xen_driver.c (xenUnifiedError): Likewise. * src/xen/xen_hypervisor.c (virXenError): Likewise. * src/xen/xen_inotify.c (virXenInotifyError): Likewise. * src/xen/xend_internal.c (virXendError): Likewise. * src/xen/xm_internal.c (xenXMError): Likewise. * src/xen/xs_internal.c (virXenStoreError): Likewise. * src/cpu/cpu.h (virCPUReportError): Use C99 rather than GNU vararg macro syntax. * src/datatypes.c (virLibConnError): Likewise. * src/interface/netcf_driver.c (interfaceReportError): Likewise. * src/libvirt.c (virLibStreamError): Likewise. * src/lxc/lxc_conf.h (lxcError): Likewise. * src/network/bridge_driver.c (networkReportError): Likewise. * src/nodeinfo.c (nodeReportError): Likewise. * src/opennebula/one_conf.h (oneError): Likewise. * src/openvz/openvz_conf.h (openvzError): Likewise. * src/phyp/phyp_driver.c (PHYP_ERROR): Likewise. * src/qemu/qemu_conf.h (qemuReportError): Likewise. * src/remote/remote_driver.c (errorf): Likewise. * src/security/security_driver.h (virSecurityReportError): Likewise. * src/test/test_driver.c (testError): Likewise. * src/uml/uml_conf.h (umlReportError): Likewise. * src/vbox/vbox_driver.c (vboxError): Likewise. * src/vbox/vbox_tmpl.c (vboxError): Likewise.
228 lines
5.7 KiB
C
228 lines
5.7 KiB
C
/*
|
|
* Copyright (C) 2009-2010 Red Hat, 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, write to the Free Software
|
|
* Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
|
*
|
|
* Authors:
|
|
* Daniel P. Berrange <berrange@redhat.com>
|
|
*/
|
|
|
|
#include <config.h>
|
|
|
|
#include <dirent.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 "hostusb.h"
|
|
#include "logging.h"
|
|
#include "memory.h"
|
|
#include "util.h"
|
|
#include "virterror_internal.h"
|
|
|
|
#define USB_SYSFS "/sys/bus/usb"
|
|
#define USB_DEVFS "/dev/bus/usb/"
|
|
#define USB_ID_LEN 10 /* "1234 5678" */
|
|
#define USB_ADDR_LEN 8 /* "123:456" */
|
|
|
|
struct _usbDevice {
|
|
unsigned bus;
|
|
unsigned dev;
|
|
|
|
char name[USB_ADDR_LEN]; /* domain:bus:slot.function */
|
|
char id[USB_ID_LEN]; /* product vendor */
|
|
char path[PATH_MAX];
|
|
};
|
|
|
|
/* For virReportOOMError() and virReportSystemError() */
|
|
#define VIR_FROM_THIS VIR_FROM_NONE
|
|
|
|
#define usbReportError(code, ...) \
|
|
virReportErrorHelper(NULL, VIR_FROM_NONE, code, __FILE__, \
|
|
__FUNCTION__, __LINE__, __VA_ARGS__)
|
|
|
|
static int usbSysReadFile(const char *f_name, const char *d_name,
|
|
int base, unsigned *value)
|
|
{
|
|
int ret = -1, tmp;
|
|
char *buf = NULL;
|
|
char *filename = NULL;
|
|
char *ignore = NULL;
|
|
|
|
tmp = virAsprintf(&filename, USB_SYSFS "/devices/%s/%s", d_name, f_name);
|
|
if (tmp < 0) {
|
|
virReportOOMError();
|
|
goto cleanup;
|
|
}
|
|
|
|
if (virFileReadAll(filename, 1024, &buf) < 0)
|
|
goto cleanup;
|
|
|
|
if (virStrToLong_ui(buf, &ignore, base, value) < 0) {
|
|
usbReportError(VIR_ERR_INTERNAL_ERROR,
|
|
_("Could not parse usb file %s"), filename);
|
|
goto cleanup;
|
|
}
|
|
|
|
ret = 0;
|
|
cleanup:
|
|
VIR_FREE(filename);
|
|
VIR_FREE(buf);
|
|
return ret;
|
|
}
|
|
|
|
static int usbFindBusByVendor(unsigned vendor, unsigned product,
|
|
unsigned *bus, unsigned *devno)
|
|
{
|
|
DIR *dir = NULL;
|
|
int ret = -1, found = 0;
|
|
char *ignore = NULL;
|
|
struct dirent *de;
|
|
|
|
dir = opendir(USB_SYSFS "/devices");
|
|
if (!dir) {
|
|
virReportSystemError(errno,
|
|
_("Could not open directory %s"),
|
|
USB_SYSFS "/devices");
|
|
goto cleanup;
|
|
}
|
|
|
|
while ((de = readdir(dir))) {
|
|
unsigned found_prod, found_vend;
|
|
if (de->d_name[0] == '.' || strchr(de->d_name, ':'))
|
|
continue;
|
|
|
|
if (usbSysReadFile("idVendor", de->d_name,
|
|
16, &found_vend) < 0)
|
|
goto cleanup;
|
|
if (usbSysReadFile("idProduct", de->d_name,
|
|
16, &found_prod) < 0)
|
|
goto cleanup;
|
|
|
|
if (found_prod == product && found_vend == vendor) {
|
|
/* Lookup bus.addr info */
|
|
char *tmpstr = de->d_name;
|
|
unsigned found_bus, found_addr;
|
|
|
|
if (STREQ(de->d_name, "usb"))
|
|
tmpstr += 3;
|
|
|
|
if (virStrToLong_ui(tmpstr, &ignore, 10, &found_bus) < 0) {
|
|
usbReportError(VIR_ERR_INTERNAL_ERROR,
|
|
_("Failed to parse dir name '%s'"),
|
|
de->d_name);
|
|
goto cleanup;
|
|
}
|
|
|
|
if (usbSysReadFile("devnum", de->d_name,
|
|
10, &found_addr) < 0)
|
|
goto cleanup;
|
|
|
|
*bus = found_bus;
|
|
*devno = found_addr;
|
|
found = 1;
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (!found)
|
|
usbReportError(VIR_ERR_INTERNAL_ERROR,
|
|
_("Did not find USB device %x:%x"), vendor, product);
|
|
else
|
|
ret = 0;
|
|
|
|
cleanup:
|
|
if (dir) {
|
|
int saved_errno = errno;
|
|
closedir (dir);
|
|
errno = saved_errno;
|
|
}
|
|
return ret;
|
|
}
|
|
|
|
usbDevice *
|
|
usbGetDevice(unsigned bus,
|
|
unsigned devno)
|
|
{
|
|
usbDevice *dev;
|
|
|
|
if (VIR_ALLOC(dev) < 0) {
|
|
virReportOOMError();
|
|
return NULL;
|
|
}
|
|
|
|
dev->bus = bus;
|
|
dev->dev = devno;
|
|
|
|
snprintf(dev->name, sizeof(dev->name), "%.3o:%.3o",
|
|
dev->bus, dev->dev);
|
|
snprintf(dev->path, sizeof(dev->path),
|
|
USB_DEVFS "%03d/%03d", dev->bus, dev->dev);
|
|
|
|
/* XXX fixme. this should be product/vendor */
|
|
snprintf(dev->id, sizeof(dev->id), "%d %d", dev->bus, dev->dev);
|
|
|
|
VIR_DEBUG("%s %s: initialized", dev->id, dev->name);
|
|
|
|
return dev;
|
|
}
|
|
|
|
|
|
usbDevice *
|
|
usbFindDevice(unsigned vendor,
|
|
unsigned product)
|
|
{
|
|
unsigned bus = 0, devno = 0;
|
|
|
|
if (usbFindBusByVendor(vendor, product, &bus, &devno) < 0) {
|
|
return NULL;
|
|
}
|
|
|
|
return usbGetDevice(bus, devno);
|
|
}
|
|
|
|
|
|
void
|
|
usbFreeDevice(usbDevice *dev)
|
|
{
|
|
VIR_DEBUG("%s %s: freeing", dev->id, dev->name);
|
|
VIR_FREE(dev);
|
|
}
|
|
|
|
|
|
unsigned usbDeviceGetBus(usbDevice *dev)
|
|
{
|
|
return dev->bus;
|
|
}
|
|
|
|
|
|
unsigned usbDeviceGetDevno(usbDevice *dev)
|
|
{
|
|
return dev->dev;
|
|
}
|
|
|
|
|
|
int usbDeviceFileIterate(usbDevice *dev,
|
|
usbDeviceFileActor actor,
|
|
void *opaque)
|
|
{
|
|
return (actor)(dev, dev->path, opaque);
|
|
}
|