2014-04-12 19:37:53 +00:00
|
|
|
/*
|
|
|
|
* bhyve_device.c: bhyve device management
|
|
|
|
*
|
|
|
|
* Copyright (C) 2014 Roman Bogorodskiy
|
|
|
|
*
|
|
|
|
* 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 "bhyve_device.h"
|
|
|
|
#include "domain_addr.h"
|
|
|
|
#include "viralloc.h"
|
|
|
|
#include "virlog.h"
|
|
|
|
#include "virstring.h"
|
|
|
|
|
|
|
|
#define VIR_FROM_THIS VIR_FROM_BHYVE
|
|
|
|
|
|
|
|
VIR_LOG_INIT("bhyve.bhyve_device");
|
|
|
|
|
|
|
|
static int
|
2021-03-11 07:16:13 +00:00
|
|
|
bhyveCollectPCIAddress(virDomainDef *def G_GNUC_UNUSED,
|
|
|
|
virDomainDeviceDef *device G_GNUC_UNUSED,
|
|
|
|
virDomainDeviceInfo *info,
|
2014-04-12 19:37:53 +00:00
|
|
|
void *opaque)
|
|
|
|
{
|
bhyve: fix SATA address allocation
As bhyve for a long time didn't have a notion of the explicit SATA
controller and created a controller for each drive, the bhyve driver
in libvirt acted in a similar way and didn't care about the SATA
controllers and assigned PCI addresses to drives directly, as
the generated command will look like this anyway:
2:0,ahci-hd,somedisk.img
This no longer makes sense because:
1. After commit c07d1c1c4f it's not possible to assign
PCI addresses to disks
2. Bhyve now supports multiple disk drives for a controller,
so it's going away from 1:1 controller:disk mapping, so
the controller object starts to make more sense now
So, this patch does the following:
- Assign PCI address to SATA controllers (previously we didn't do this)
- Assign disk addresses instead of PCI addresses for disks. Now, when
building a bhyve command, we take PCI address not from the disk
itself but from its controller
- Assign addresses at XML parsing time using the
assignAddressesCallback. This is done mainly for being able to
verify address allocation via xml2xml tests
- Adjust existing bhyvexml2{xml,argv} tests to chase the new
address allocation
This patch is largely based on work of Fabian Freyer.
2017-01-05 12:51:25 +00:00
|
|
|
if (info->type == VIR_DOMAIN_DEVICE_ADDRESS_TYPE_DRIVE)
|
|
|
|
return 0;
|
|
|
|
|
2021-03-11 07:16:13 +00:00
|
|
|
virDomainPCIAddressSet *addrs = opaque;
|
|
|
|
virPCIDeviceAddress *addr = &info->addr.pci;
|
2014-04-12 19:37:53 +00:00
|
|
|
|
2020-09-20 14:53:57 +00:00
|
|
|
if (addr->domain == 0 && addr->bus == 0 && addr->slot == 0) {
|
2014-04-12 19:37:53 +00:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2016-10-23 16:14:16 +00:00
|
|
|
if (virDomainPCIAddressReserveAddr(addrs, addr,
|
2017-06-15 08:38:33 +00:00
|
|
|
VIR_PCI_CONNECT_TYPE_PCI_DEVICE, 0) < 0) {
|
2019-10-21 18:19:11 +00:00
|
|
|
return -1;
|
2016-10-23 16:14:16 +00:00
|
|
|
}
|
2014-04-12 19:37:53 +00:00
|
|
|
|
2019-10-21 18:19:11 +00:00
|
|
|
return 0;
|
2014-04-12 19:37:53 +00:00
|
|
|
}
|
|
|
|
|
2021-03-11 07:16:13 +00:00
|
|
|
virDomainPCIAddressSet *
|
|
|
|
bhyveDomainPCIAddressSetCreate(virDomainDef *def, unsigned int nbuses)
|
2014-04-12 19:37:53 +00:00
|
|
|
{
|
2021-03-11 07:16:13 +00:00
|
|
|
virDomainPCIAddressSet *addrs;
|
2014-04-12 19:37:53 +00:00
|
|
|
|
2018-11-08 11:00:24 +00:00
|
|
|
if ((addrs = virDomainPCIAddressSetAlloc(nbuses,
|
|
|
|
VIR_PCI_ADDRESS_EXTENSION_NONE)) == NULL)
|
2014-04-12 19:37:53 +00:00
|
|
|
return NULL;
|
|
|
|
|
|
|
|
if (virDomainPCIAddressBusSetModel(&addrs->buses[0],
|
2020-04-16 10:39:49 +00:00
|
|
|
VIR_DOMAIN_CONTROLLER_MODEL_PCI_ROOT,
|
|
|
|
true) < 0)
|
2014-04-12 19:37:53 +00:00
|
|
|
goto error;
|
|
|
|
|
|
|
|
if (virDomainDeviceInfoIterate(def, bhyveCollectPCIAddress, addrs) < 0)
|
|
|
|
goto error;
|
|
|
|
|
|
|
|
return addrs;
|
|
|
|
|
|
|
|
error:
|
|
|
|
virDomainPCIAddressSetFree(addrs);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int
|
2021-03-11 07:16:13 +00:00
|
|
|
bhyveAssignDevicePCISlots(virDomainDef *def,
|
|
|
|
virDomainPCIAddressSet *addrs)
|
2014-04-12 19:37:53 +00:00
|
|
|
{
|
|
|
|
size_t i;
|
2022-02-11 13:30:09 +00:00
|
|
|
virPCIDeviceAddress lpc_addr = { .slot = 0x1 };
|
2014-04-12 19:37:53 +00:00
|
|
|
|
2020-09-20 14:53:57 +00:00
|
|
|
/* If the user didn't explicitly specify slot 1 for some of the devices,
|
|
|
|
reserve it for LPC, even if there's no LPC device configured.
|
|
|
|
If the slot 1 is used by some other device, LPC will have an address
|
|
|
|
auto-assigned.
|
|
|
|
|
|
|
|
The idea behind that is to try to use slot 1 for the LPC device unless
|
|
|
|
user specifically configured otherwise.*/
|
|
|
|
if (!virDomainPCIAddressSlotInUse(addrs, &lpc_addr)) {
|
|
|
|
if (virDomainPCIAddressReserveAddr(addrs, &lpc_addr,
|
|
|
|
VIR_PCI_CONNECT_TYPE_PCI_DEVICE, 0) < 0) {
|
|
|
|
return -1;
|
|
|
|
}
|
2014-04-12 19:37:53 +00:00
|
|
|
|
2020-09-20 14:53:57 +00:00
|
|
|
for (i = 0; i < def->ncontrollers; i++) {
|
|
|
|
if ((def->controllers[i]->type == VIR_DOMAIN_CONTROLLER_TYPE_ISA) &&
|
|
|
|
virDeviceInfoPCIAddressIsWanted(&def->controllers[i]->info)) {
|
|
|
|
def->controllers[i]->info.type = VIR_DOMAIN_DEVICE_ADDRESS_TYPE_PCI;
|
|
|
|
def->controllers[i]->info.addr.pci = lpc_addr;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2019-02-17 13:04:00 +00:00
|
|
|
}
|
|
|
|
|
2014-04-12 19:37:53 +00:00
|
|
|
for (i = 0; i < def->ncontrollers; i++) {
|
bhyve: fix SATA address allocation
As bhyve for a long time didn't have a notion of the explicit SATA
controller and created a controller for each drive, the bhyve driver
in libvirt acted in a similar way and didn't care about the SATA
controllers and assigned PCI addresses to drives directly, as
the generated command will look like this anyway:
2:0,ahci-hd,somedisk.img
This no longer makes sense because:
1. After commit c07d1c1c4f it's not possible to assign
PCI addresses to disks
2. Bhyve now supports multiple disk drives for a controller,
so it's going away from 1:1 controller:disk mapping, so
the controller object starts to make more sense now
So, this patch does the following:
- Assign PCI address to SATA controllers (previously we didn't do this)
- Assign disk addresses instead of PCI addresses for disks. Now, when
building a bhyve command, we take PCI address not from the disk
itself but from its controller
- Assign addresses at XML parsing time using the
assignAddressesCallback. This is done mainly for being able to
verify address allocation via xml2xml tests
- Adjust existing bhyvexml2{xml,argv} tests to chase the new
address allocation
This patch is largely based on work of Fabian Freyer.
2017-01-05 12:51:25 +00:00
|
|
|
if ((def->controllers[i]->type == VIR_DOMAIN_CONTROLLER_TYPE_PCI) ||
|
2017-03-20 13:58:51 +00:00
|
|
|
(def->controllers[i]->type == VIR_DOMAIN_CONTROLLER_TYPE_SATA) ||
|
|
|
|
((def->controllers[i]->type == VIR_DOMAIN_CONTROLLER_TYPE_USB) &&
|
2020-09-20 14:53:57 +00:00
|
|
|
(def->controllers[i]->model == VIR_DOMAIN_CONTROLLER_MODEL_USB_NEC_XHCI)) ||
|
|
|
|
def->controllers[i]->type == VIR_DOMAIN_CONTROLLER_TYPE_ISA) {
|
2016-05-17 18:06:36 +00:00
|
|
|
if (def->controllers[i]->model == VIR_DOMAIN_CONTROLLER_MODEL_PCI_ROOT ||
|
2018-08-17 13:12:42 +00:00
|
|
|
!virDeviceInfoPCIAddressIsWanted(&def->controllers[i]->info))
|
2014-04-12 19:37:53 +00:00
|
|
|
continue;
|
|
|
|
|
2016-10-22 17:28:07 +00:00
|
|
|
if (virDomainPCIAddressReserveNextAddr(addrs,
|
2014-04-12 19:37:53 +00:00
|
|
|
&def->controllers[i]->info,
|
2016-10-22 17:28:07 +00:00
|
|
|
VIR_PCI_CONNECT_TYPE_PCI_DEVICE,
|
|
|
|
-1) < 0)
|
2020-01-06 21:57:25 +00:00
|
|
|
return -1;
|
2014-04-12 19:37:53 +00:00
|
|
|
}
|
bhyve: fix SATA address allocation
As bhyve for a long time didn't have a notion of the explicit SATA
controller and created a controller for each drive, the bhyve driver
in libvirt acted in a similar way and didn't care about the SATA
controllers and assigned PCI addresses to drives directly, as
the generated command will look like this anyway:
2:0,ahci-hd,somedisk.img
This no longer makes sense because:
1. After commit c07d1c1c4f it's not possible to assign
PCI addresses to disks
2. Bhyve now supports multiple disk drives for a controller,
so it's going away from 1:1 controller:disk mapping, so
the controller object starts to make more sense now
So, this patch does the following:
- Assign PCI address to SATA controllers (previously we didn't do this)
- Assign disk addresses instead of PCI addresses for disks. Now, when
building a bhyve command, we take PCI address not from the disk
itself but from its controller
- Assign addresses at XML parsing time using the
assignAddressesCallback. This is done mainly for being able to
verify address allocation via xml2xml tests
- Adjust existing bhyvexml2{xml,argv} tests to chase the new
address allocation
This patch is largely based on work of Fabian Freyer.
2017-01-05 12:51:25 +00:00
|
|
|
}
|
2014-04-12 19:37:53 +00:00
|
|
|
|
bhyve: fix SATA address allocation
As bhyve for a long time didn't have a notion of the explicit SATA
controller and created a controller for each drive, the bhyve driver
in libvirt acted in a similar way and didn't care about the SATA
controllers and assigned PCI addresses to drives directly, as
the generated command will look like this anyway:
2:0,ahci-hd,somedisk.img
This no longer makes sense because:
1. After commit c07d1c1c4f it's not possible to assign
PCI addresses to disks
2. Bhyve now supports multiple disk drives for a controller,
so it's going away from 1:1 controller:disk mapping, so
the controller object starts to make more sense now
So, this patch does the following:
- Assign PCI address to SATA controllers (previously we didn't do this)
- Assign disk addresses instead of PCI addresses for disks. Now, when
building a bhyve command, we take PCI address not from the disk
itself but from its controller
- Assign addresses at XML parsing time using the
assignAddressesCallback. This is done mainly for being able to
verify address allocation via xml2xml tests
- Adjust existing bhyvexml2{xml,argv} tests to chase the new
address allocation
This patch is largely based on work of Fabian Freyer.
2017-01-05 12:51:25 +00:00
|
|
|
for (i = 0; i < def->nnets; i++) {
|
2018-08-17 13:12:42 +00:00
|
|
|
if (!virDeviceInfoPCIAddressIsWanted(&def->nets[i]->info))
|
bhyve: fix SATA address allocation
As bhyve for a long time didn't have a notion of the explicit SATA
controller and created a controller for each drive, the bhyve driver
in libvirt acted in a similar way and didn't care about the SATA
controllers and assigned PCI addresses to drives directly, as
the generated command will look like this anyway:
2:0,ahci-hd,somedisk.img
This no longer makes sense because:
1. After commit c07d1c1c4f it's not possible to assign
PCI addresses to disks
2. Bhyve now supports multiple disk drives for a controller,
so it's going away from 1:1 controller:disk mapping, so
the controller object starts to make more sense now
So, this patch does the following:
- Assign PCI address to SATA controllers (previously we didn't do this)
- Assign disk addresses instead of PCI addresses for disks. Now, when
building a bhyve command, we take PCI address not from the disk
itself but from its controller
- Assign addresses at XML parsing time using the
assignAddressesCallback. This is done mainly for being able to
verify address allocation via xml2xml tests
- Adjust existing bhyvexml2{xml,argv} tests to chase the new
address allocation
This patch is largely based on work of Fabian Freyer.
2017-01-05 12:51:25 +00:00
|
|
|
continue;
|
|
|
|
if (virDomainPCIAddressReserveNextAddr(addrs,
|
|
|
|
&def->nets[i]->info,
|
|
|
|
VIR_PCI_CONNECT_TYPE_PCI_DEVICE,
|
2017-02-01 16:19:31 +00:00
|
|
|
-1) < 0)
|
2020-01-06 21:57:25 +00:00
|
|
|
return -1;
|
2017-02-01 16:19:31 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
for (i = 0; i < def->ndisks; i++) {
|
|
|
|
/* We only handle virtio disk addresses as SATA disks are
|
|
|
|
* attached to a controller and don't have their own PCI
|
|
|
|
* addresses */
|
|
|
|
if (def->disks[i]->bus != VIR_DOMAIN_DISK_BUS_VIRTIO)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
if (def->disks[i]->info.type == VIR_DOMAIN_DEVICE_ADDRESS_TYPE_PCI &&
|
|
|
|
!virPCIDeviceAddressIsEmpty(&def->disks[i]->info.addr.pci))
|
|
|
|
continue;
|
|
|
|
if (virDomainPCIAddressReserveNextAddr(addrs, &def->disks[i]->info,
|
|
|
|
VIR_PCI_CONNECT_TYPE_PCI_DEVICE,
|
bhyve: fix SATA address allocation
As bhyve for a long time didn't have a notion of the explicit SATA
controller and created a controller for each drive, the bhyve driver
in libvirt acted in a similar way and didn't care about the SATA
controllers and assigned PCI addresses to drives directly, as
the generated command will look like this anyway:
2:0,ahci-hd,somedisk.img
This no longer makes sense because:
1. After commit c07d1c1c4f it's not possible to assign
PCI addresses to disks
2. Bhyve now supports multiple disk drives for a controller,
so it's going away from 1:1 controller:disk mapping, so
the controller object starts to make more sense now
So, this patch does the following:
- Assign PCI address to SATA controllers (previously we didn't do this)
- Assign disk addresses instead of PCI addresses for disks. Now, when
building a bhyve command, we take PCI address not from the disk
itself but from its controller
- Assign addresses at XML parsing time using the
assignAddressesCallback. This is done mainly for being able to
verify address allocation via xml2xml tests
- Adjust existing bhyvexml2{xml,argv} tests to chase the new
address allocation
This patch is largely based on work of Fabian Freyer.
2017-01-05 12:51:25 +00:00
|
|
|
-1) < 0)
|
2020-01-06 21:57:25 +00:00
|
|
|
return -1;
|
2014-04-12 19:37:53 +00:00
|
|
|
}
|
|
|
|
|
2016-07-16 21:03:33 +00:00
|
|
|
for (i = 0; i < def->nvideos; i++) {
|
2018-08-17 13:12:42 +00:00
|
|
|
if (!virDeviceInfoPCIAddressIsWanted(&def->videos[i]->info))
|
2016-07-16 21:03:33 +00:00
|
|
|
continue;
|
|
|
|
if (virDomainPCIAddressReserveNextAddr(addrs,
|
|
|
|
&def->videos[i]->info,
|
|
|
|
VIR_PCI_CONNECT_TYPE_PCI_DEVICE,
|
|
|
|
-1) < 0)
|
2020-01-06 21:57:25 +00:00
|
|
|
return -1;
|
2016-07-16 21:03:33 +00:00
|
|
|
}
|
|
|
|
|
2020-07-14 14:44:13 +00:00
|
|
|
for (i = 0; i < def->nsounds; i++) {
|
|
|
|
if (!virDeviceInfoPCIAddressIsWanted(&def->sounds[i]->info))
|
|
|
|
continue;
|
|
|
|
if (virDomainPCIAddressReserveNextAddr(addrs,
|
|
|
|
&def->sounds[i]->info,
|
|
|
|
VIR_PCI_CONNECT_TYPE_PCI_DEVICE,
|
|
|
|
-1) < 0)
|
|
|
|
return -1;
|
|
|
|
}
|
2016-07-16 21:03:33 +00:00
|
|
|
|
2020-10-07 16:07:04 +00:00
|
|
|
for (i = 0; i < def->nfss; i++) {
|
|
|
|
if (!virDeviceInfoPCIAddressIsWanted(&def->fss[i]->info))
|
|
|
|
continue;
|
|
|
|
if (virDomainPCIAddressReserveNextAddr(addrs,
|
|
|
|
&def->fss[i]->info,
|
|
|
|
VIR_PCI_CONNECT_TYPE_PCI_DEVICE,
|
|
|
|
-1) < 0)
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2014-04-12 19:37:53 +00:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2021-03-11 07:16:13 +00:00
|
|
|
int bhyveDomainAssignPCIAddresses(virDomainDef *def,
|
|
|
|
virDomainObj *obj)
|
2014-04-12 19:37:53 +00:00
|
|
|
{
|
2021-03-11 07:16:13 +00:00
|
|
|
virDomainPCIAddressSet *addrs = NULL;
|
|
|
|
bhyveDomainObjPrivate *priv = NULL;
|
2014-04-12 19:37:53 +00:00
|
|
|
|
|
|
|
if (!(addrs = bhyveDomainPCIAddressSetCreate(def, 1)))
|
2019-10-21 18:19:11 +00:00
|
|
|
return -1;
|
2014-04-12 19:37:53 +00:00
|
|
|
|
|
|
|
if (bhyveAssignDevicePCISlots(def, addrs) < 0)
|
2019-10-21 18:19:11 +00:00
|
|
|
return -1;
|
2014-04-12 19:37:53 +00:00
|
|
|
|
|
|
|
if (obj && obj->privateData) {
|
|
|
|
priv = obj->privateData;
|
|
|
|
if (addrs) {
|
|
|
|
virDomainPCIAddressSetFree(priv->pciaddrs);
|
|
|
|
priv->persistentAddrs = 1;
|
|
|
|
priv->pciaddrs = addrs;
|
|
|
|
} else {
|
|
|
|
priv->persistentAddrs = 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-10-21 18:19:11 +00:00
|
|
|
return 0;
|
2014-04-12 19:37:53 +00:00
|
|
|
}
|
|
|
|
|
2021-03-11 07:16:13 +00:00
|
|
|
int bhyveDomainAssignAddresses(virDomainDef *def, virDomainObj *obj)
|
2014-04-12 19:37:53 +00:00
|
|
|
{
|
|
|
|
return bhyveDomainAssignPCIAddresses(def, obj);
|
|
|
|
}
|