mirror of
https://gitlab.com/libvirt/libvirt.git
synced 2025-01-10 23:07:44 +00:00
conf: eliminate concept of "reserveEntireSlot"
setting reserveEntireSlot really accomplishes nothing - instead of going to the trouble of computing the value for reserveEntireSlot and then possibly setting *all* functions of the slot as in-use, we can just set the in-use bit only for the specific function being used by a device. Later we will know from the context (the PCI connect flags, and whether we are reserving a specific address or asking for "the next available") whether or not it is okay to allocate other functions on the same slot. Although it's not used yet, we allow specifying "-1" for the function number when looking for the "next available slot" - this is going to end up meaning "return the lowest available function in the slot, but since we currently only provide a function from an otherwise unused slot, "-1" ends up meaning "0".
This commit is contained in:
parent
9838cad9cd
commit
9ff9d9f5a9
@ -527,11 +527,9 @@ virDomainPCIAddressSlotInUse(virDomainPCIAddressSetPtr addrs,
|
|||||||
|
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Reserve a slot (or just one function) for a device. If
|
* Reserve a function in a slot. If fromConfig is true, the address
|
||||||
* reserveEntireSlot is true, all functions for the slot are reserved,
|
* being requested came directly from the config and errors should be
|
||||||
* otherwise only one. If fromConfig is true, the address being
|
* worded appropriately. If fromConfig is false, the address was
|
||||||
* requested came directly from the config and errors should be worded
|
|
||||||
* appropriately. If fromConfig is false, the address was
|
|
||||||
* automatically created by libvirt, so it is an internal error (not
|
* automatically created by libvirt, so it is an internal error (not
|
||||||
* XML).
|
* XML).
|
||||||
*/
|
*/
|
||||||
@ -539,7 +537,6 @@ int
|
|||||||
virDomainPCIAddressReserveAddr(virDomainPCIAddressSetPtr addrs,
|
virDomainPCIAddressReserveAddr(virDomainPCIAddressSetPtr addrs,
|
||||||
virPCIDeviceAddressPtr addr,
|
virPCIDeviceAddressPtr addr,
|
||||||
virDomainPCIConnectFlags flags,
|
virDomainPCIConnectFlags flags,
|
||||||
bool reserveEntireSlot,
|
|
||||||
bool fromConfig)
|
bool fromConfig)
|
||||||
{
|
{
|
||||||
int ret = -1;
|
int ret = -1;
|
||||||
@ -562,33 +559,13 @@ virDomainPCIAddressReserveAddr(virDomainPCIAddressSetPtr addrs,
|
|||||||
|
|
||||||
bus = &addrs->buses[addr->bus];
|
bus = &addrs->buses[addr->bus];
|
||||||
|
|
||||||
if (reserveEntireSlot) {
|
if (bus->slot[addr->slot].functions & (1 << addr->function)) {
|
||||||
if (bus->slot[addr->slot].functions) {
|
virReportError(errType,
|
||||||
virReportError(errType,
|
_("Attempted double use of PCI Address %s"), addrStr);
|
||||||
_("Attempted double use of PCI slot %s "
|
goto cleanup;
|
||||||
"(may need \"multifunction='on'\" for "
|
|
||||||
"device on function 0)"), addrStr);
|
|
||||||
goto cleanup;
|
|
||||||
}
|
|
||||||
bus->slot[addr->slot].functions = 0xFF; /* reserve all functions of slot */
|
|
||||||
VIR_DEBUG("Reserving PCI slot %s (multifunction='off')", addrStr);
|
|
||||||
} else {
|
|
||||||
if (bus->slot[addr->slot].functions & (1 << addr->function)) {
|
|
||||||
if (addr->function == 0) {
|
|
||||||
virReportError(errType,
|
|
||||||
_("Attempted double use of PCI Address %s"),
|
|
||||||
addrStr);
|
|
||||||
} else {
|
|
||||||
virReportError(errType,
|
|
||||||
_("Attempted double use of PCI Address %s "
|
|
||||||
"(may need \"multifunction='on'\" "
|
|
||||||
"for device on function 0)"), addrStr);
|
|
||||||
}
|
|
||||||
goto cleanup;
|
|
||||||
}
|
|
||||||
bus->slot[addr->slot].functions |= (1 << addr->function);
|
|
||||||
VIR_DEBUG("Reserving PCI address %s", addrStr);
|
|
||||||
}
|
}
|
||||||
|
bus->slot[addr->slot].functions |= (1 << addr->function);
|
||||||
|
VIR_DEBUG("Reserving PCI address %s", addrStr);
|
||||||
|
|
||||||
ret = 0;
|
ret = 0;
|
||||||
cleanup:
|
cleanup:
|
||||||
@ -602,7 +579,7 @@ virDomainPCIAddressReserveSlot(virDomainPCIAddressSetPtr addrs,
|
|||||||
virPCIDeviceAddressPtr addr,
|
virPCIDeviceAddressPtr addr,
|
||||||
virDomainPCIConnectFlags flags)
|
virDomainPCIConnectFlags flags)
|
||||||
{
|
{
|
||||||
return virDomainPCIAddressReserveAddr(addrs, addr, flags, true, false);
|
return virDomainPCIAddressReserveAddr(addrs, addr, flags, false);
|
||||||
}
|
}
|
||||||
|
|
||||||
int
|
int
|
||||||
@ -637,8 +614,8 @@ virDomainPCIAddressEnsureAddr(virDomainPCIAddressSetPtr addrs,
|
|||||||
addrStr, flags, true))
|
addrStr, flags, true))
|
||||||
goto cleanup;
|
goto cleanup;
|
||||||
|
|
||||||
ret = virDomainPCIAddressReserveAddr(addrs, &dev->addr.pci, flags,
|
ret = virDomainPCIAddressReserveAddr(addrs, &dev->addr.pci,
|
||||||
true, true);
|
flags, true);
|
||||||
} else {
|
} else {
|
||||||
ret = virDomainPCIAddressReserveNextSlot(addrs, dev, flags);
|
ret = virDomainPCIAddressReserveNextSlot(addrs, dev, flags);
|
||||||
}
|
}
|
||||||
@ -716,6 +693,7 @@ virDomainPCIAddressSetFree(virDomainPCIAddressSetPtr addrs)
|
|||||||
static int ATTRIBUTE_NONNULL(1) ATTRIBUTE_NONNULL(2)
|
static int ATTRIBUTE_NONNULL(1) ATTRIBUTE_NONNULL(2)
|
||||||
virDomainPCIAddressGetNextSlot(virDomainPCIAddressSetPtr addrs,
|
virDomainPCIAddressGetNextSlot(virDomainPCIAddressSetPtr addrs,
|
||||||
virPCIDeviceAddressPtr next_addr,
|
virPCIDeviceAddressPtr next_addr,
|
||||||
|
int function,
|
||||||
virDomainPCIConnectFlags flags)
|
virDomainPCIConnectFlags flags)
|
||||||
{
|
{
|
||||||
/* default to starting the search for a free slot from
|
/* default to starting the search for a free slot from
|
||||||
@ -743,6 +721,12 @@ virDomainPCIAddressGetNextSlot(virDomainPCIAddressSetPtr addrs,
|
|||||||
a.slot = addrs->buses[0].minSlot;
|
a.slot = addrs->buses[0].minSlot;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/* if the caller asks for "any function", give them function 0 */
|
||||||
|
if (function == -1)
|
||||||
|
a.function = 0;
|
||||||
|
else
|
||||||
|
a.function = function;
|
||||||
|
|
||||||
while (a.bus < addrs->nbuses) {
|
while (a.bus < addrs->nbuses) {
|
||||||
VIR_FREE(addrStr);
|
VIR_FREE(addrStr);
|
||||||
if (!(addrStr = virDomainPCIAddressAsString(&a)))
|
if (!(addrStr = virDomainPCIAddressAsString(&a)))
|
||||||
@ -821,14 +805,13 @@ virDomainPCIAddressGetNextSlot(virDomainPCIAddressSetPtr addrs,
|
|||||||
* @dev: virDomainDeviceInfo that should get the new address.
|
* @dev: virDomainDeviceInfo that should get the new address.
|
||||||
* @flags: CONNECT_TYPE flags for the device that needs an address.
|
* @flags: CONNECT_TYPE flags for the device that needs an address.
|
||||||
* @function: which function on the slot to mark as reserved
|
* @function: which function on the slot to mark as reserved
|
||||||
* (if @reserveEntireSlot is false)
|
|
||||||
* @reserveEntireSlot: true to reserve all functions on the new slot,
|
|
||||||
* false to reserve just @function
|
|
||||||
*
|
*
|
||||||
* Find the next *completely unreserved* slot with compatible
|
* Find the next *completely unreserved* slot with compatible
|
||||||
* connection @flags, mark either one function or the entire
|
* connection @flags, mark one function of the slot as in-use
|
||||||
* slot as in-use (according to @function and @reserveEntireSlot),
|
* (according to @function), then set @dev->addr.pci with this newly
|
||||||
* and set @dev->addr.pci with this newly reserved address.
|
* reserved address. If @function is -1, then the lowest unused
|
||||||
|
* function of the slot will be reserved (and since we only look for
|
||||||
|
* completely unused slots, that means "0").
|
||||||
*
|
*
|
||||||
* returns 0 on success, or -1 on failure.
|
* returns 0 on success, or -1 on failure.
|
||||||
*/
|
*/
|
||||||
@ -836,17 +819,14 @@ int
|
|||||||
virDomainPCIAddressReserveNextAddr(virDomainPCIAddressSetPtr addrs,
|
virDomainPCIAddressReserveNextAddr(virDomainPCIAddressSetPtr addrs,
|
||||||
virDomainDeviceInfoPtr dev,
|
virDomainDeviceInfoPtr dev,
|
||||||
virDomainPCIConnectFlags flags,
|
virDomainPCIConnectFlags flags,
|
||||||
unsigned int function,
|
int function)
|
||||||
bool reserveEntireSlot)
|
|
||||||
{
|
{
|
||||||
virPCIDeviceAddress addr;
|
virPCIDeviceAddress addr;
|
||||||
|
|
||||||
if (virDomainPCIAddressGetNextSlot(addrs, &addr, flags) < 0)
|
if (virDomainPCIAddressGetNextSlot(addrs, &addr, function, flags) < 0)
|
||||||
return -1;
|
return -1;
|
||||||
|
|
||||||
addr.function = reserveEntireSlot ? 0 : function;
|
if (virDomainPCIAddressReserveAddr(addrs, &addr, flags, false) < 0)
|
||||||
|
|
||||||
if (virDomainPCIAddressReserveAddr(addrs, &addr, flags, reserveEntireSlot, false) < 0)
|
|
||||||
return -1;
|
return -1;
|
||||||
|
|
||||||
addrs->lastaddr = addr;
|
addrs->lastaddr = addr;
|
||||||
@ -866,7 +846,7 @@ virDomainPCIAddressReserveNextSlot(virDomainPCIAddressSetPtr addrs,
|
|||||||
virDomainDeviceInfoPtr dev,
|
virDomainDeviceInfoPtr dev,
|
||||||
virDomainPCIConnectFlags flags)
|
virDomainPCIConnectFlags flags)
|
||||||
{
|
{
|
||||||
return virDomainPCIAddressReserveNextAddr(addrs, dev, flags, 0, true);
|
return virDomainPCIAddressReserveNextAddr(addrs, dev, flags, 0);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@ -141,7 +141,6 @@ int virDomainPCIAddressSetGrow(virDomainPCIAddressSetPtr addrs,
|
|||||||
int virDomainPCIAddressReserveAddr(virDomainPCIAddressSetPtr addrs,
|
int virDomainPCIAddressReserveAddr(virDomainPCIAddressSetPtr addrs,
|
||||||
virPCIDeviceAddressPtr addr,
|
virPCIDeviceAddressPtr addr,
|
||||||
virDomainPCIConnectFlags flags,
|
virDomainPCIConnectFlags flags,
|
||||||
bool reserveEntireSlot,
|
|
||||||
bool fromConfig)
|
bool fromConfig)
|
||||||
ATTRIBUTE_NONNULL(1) ATTRIBUTE_NONNULL(2);
|
ATTRIBUTE_NONNULL(1) ATTRIBUTE_NONNULL(2);
|
||||||
|
|
||||||
@ -166,8 +165,7 @@ int virDomainPCIAddressReleaseSlot(virDomainPCIAddressSetPtr addrs,
|
|||||||
int virDomainPCIAddressReserveNextAddr(virDomainPCIAddressSetPtr addrs,
|
int virDomainPCIAddressReserveNextAddr(virDomainPCIAddressSetPtr addrs,
|
||||||
virDomainDeviceInfoPtr dev,
|
virDomainDeviceInfoPtr dev,
|
||||||
virDomainPCIConnectFlags flags,
|
virDomainPCIConnectFlags flags,
|
||||||
unsigned int function,
|
int function)
|
||||||
bool reserveEntireSlot)
|
|
||||||
ATTRIBUTE_NONNULL(1) ATTRIBUTE_NONNULL(2);
|
ATTRIBUTE_NONNULL(1) ATTRIBUTE_NONNULL(2);
|
||||||
|
|
||||||
int virDomainPCIAddressReserveNextSlot(virDomainPCIAddressSetPtr addrs,
|
int virDomainPCIAddressReserveNextSlot(virDomainPCIAddressSetPtr addrs,
|
||||||
|
@ -937,12 +937,11 @@ qemuDomainFillDevicePCIConnectFlags(virDomainDefPtr def,
|
|||||||
static int
|
static int
|
||||||
qemuDomainPCIAddressReserveNextAddr(virDomainPCIAddressSetPtr addrs,
|
qemuDomainPCIAddressReserveNextAddr(virDomainPCIAddressSetPtr addrs,
|
||||||
virDomainDeviceInfoPtr dev,
|
virDomainDeviceInfoPtr dev,
|
||||||
unsigned int function,
|
unsigned int function)
|
||||||
bool reserveEntireSlot)
|
|
||||||
{
|
{
|
||||||
return virDomainPCIAddressReserveNextAddr(addrs, dev,
|
return virDomainPCIAddressReserveNextAddr(addrs, dev,
|
||||||
dev->pciConnectFlags,
|
dev->pciConnectFlags,
|
||||||
function, reserveEntireSlot);
|
function);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@ -950,7 +949,7 @@ static int
|
|||||||
qemuDomainPCIAddressReserveNextSlot(virDomainPCIAddressSetPtr addrs,
|
qemuDomainPCIAddressReserveNextSlot(virDomainPCIAddressSetPtr addrs,
|
||||||
virDomainDeviceInfoPtr dev)
|
virDomainDeviceInfoPtr dev)
|
||||||
{
|
{
|
||||||
return qemuDomainPCIAddressReserveNextAddr(addrs, dev, 0, true);
|
return qemuDomainPCIAddressReserveNextAddr(addrs, dev, 0);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@ -963,7 +962,6 @@ qemuDomainCollectPCIAddress(virDomainDefPtr def ATTRIBUTE_UNUSED,
|
|||||||
virDomainPCIAddressSetPtr addrs = opaque;
|
virDomainPCIAddressSetPtr addrs = opaque;
|
||||||
int ret = -1;
|
int ret = -1;
|
||||||
virPCIDeviceAddressPtr addr = &info->addr.pci;
|
virPCIDeviceAddressPtr addr = &info->addr.pci;
|
||||||
bool entireSlot;
|
|
||||||
|
|
||||||
if (!virDeviceInfoPCIAddressPresent(info) ||
|
if (!virDeviceInfoPCIAddressPresent(info) ||
|
||||||
((device->type == VIR_DOMAIN_DEVICE_HOSTDEV) &&
|
((device->type == VIR_DOMAIN_DEVICE_HOSTDEV) &&
|
||||||
@ -1036,12 +1034,10 @@ qemuDomainCollectPCIAddress(virDomainDefPtr def ATTRIBUTE_UNUSED,
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
entireSlot = (addr->function == 0 &&
|
if (virDomainPCIAddressReserveAddr(addrs, addr,
|
||||||
addr->multi != VIR_TRISTATE_SWITCH_ON);
|
info->pciConnectFlags, true) < 0) {
|
||||||
|
|
||||||
if (virDomainPCIAddressReserveAddr(addrs, addr, info->pciConnectFlags,
|
|
||||||
entireSlot, true) < 0)
|
|
||||||
goto cleanup;
|
goto cleanup;
|
||||||
|
}
|
||||||
|
|
||||||
ret = 0;
|
ret = 0;
|
||||||
cleanup:
|
cleanup:
|
||||||
@ -1329,7 +1325,7 @@ qemuDomainValidateDevicePCISlotsQ35(virDomainDefPtr def,
|
|||||||
}
|
}
|
||||||
if (assign) {
|
if (assign) {
|
||||||
if (virDomainPCIAddressReserveAddr(addrs, &tmp_addr,
|
if (virDomainPCIAddressReserveAddr(addrs, &tmp_addr,
|
||||||
flags, false, true) < 0)
|
flags, true) < 0)
|
||||||
goto cleanup;
|
goto cleanup;
|
||||||
cont->info.type = VIR_DOMAIN_DEVICE_ADDRESS_TYPE_PCI;
|
cont->info.type = VIR_DOMAIN_DEVICE_ADDRESS_TYPE_PCI;
|
||||||
cont->info.addr.pci.domain = 0;
|
cont->info.addr.pci.domain = 0;
|
||||||
@ -1352,7 +1348,7 @@ qemuDomainValidateDevicePCISlotsQ35(virDomainDefPtr def,
|
|||||||
tmp_addr.slot = 0x1E;
|
tmp_addr.slot = 0x1E;
|
||||||
if (!virDomainPCIAddressSlotInUse(addrs, &tmp_addr)) {
|
if (!virDomainPCIAddressSlotInUse(addrs, &tmp_addr)) {
|
||||||
if (virDomainPCIAddressReserveAddr(addrs, &tmp_addr,
|
if (virDomainPCIAddressReserveAddr(addrs, &tmp_addr,
|
||||||
flags, true, false) < 0)
|
flags, false) < 0)
|
||||||
goto cleanup;
|
goto cleanup;
|
||||||
cont->info.type = VIR_DOMAIN_DEVICE_ADDRESS_TYPE_PCI;
|
cont->info.type = VIR_DOMAIN_DEVICE_ADDRESS_TYPE_PCI;
|
||||||
cont->info.addr.pci.domain = 0;
|
cont->info.addr.pci.domain = 0;
|
||||||
@ -1375,13 +1371,13 @@ qemuDomainValidateDevicePCISlotsQ35(virDomainDefPtr def,
|
|||||||
tmp_addr.slot = 0x1F;
|
tmp_addr.slot = 0x1F;
|
||||||
tmp_addr.function = 0;
|
tmp_addr.function = 0;
|
||||||
tmp_addr.multi = VIR_TRISTATE_SWITCH_ON;
|
tmp_addr.multi = VIR_TRISTATE_SWITCH_ON;
|
||||||
if (virDomainPCIAddressReserveAddr(addrs, &tmp_addr, flags,
|
if (virDomainPCIAddressReserveAddr(addrs, &tmp_addr,
|
||||||
false, false) < 0)
|
flags, false) < 0)
|
||||||
goto cleanup;
|
goto cleanup;
|
||||||
tmp_addr.function = 3;
|
tmp_addr.function = 3;
|
||||||
tmp_addr.multi = VIR_TRISTATE_SWITCH_ABSENT;
|
tmp_addr.multi = VIR_TRISTATE_SWITCH_ABSENT;
|
||||||
if (virDomainPCIAddressReserveAddr(addrs, &tmp_addr, flags,
|
if (virDomainPCIAddressReserveAddr(addrs, &tmp_addr,
|
||||||
false, false) < 0)
|
flags, false) < 0)
|
||||||
goto cleanup;
|
goto cleanup;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -1681,7 +1677,7 @@ qemuDomainAssignDevicePCISlots(virDomainDefPtr def,
|
|||||||
/* Reserve this function on the slot we found */
|
/* Reserve this function on the slot we found */
|
||||||
if (virDomainPCIAddressReserveAddr(addrs, &addr,
|
if (virDomainPCIAddressReserveAddr(addrs, &addr,
|
||||||
cont->info.pciConnectFlags,
|
cont->info.pciConnectFlags,
|
||||||
false, true) < 0)
|
true) < 0)
|
||||||
goto error;
|
goto error;
|
||||||
|
|
||||||
cont->info.type = VIR_DOMAIN_DEVICE_ADDRESS_TYPE_PCI;
|
cont->info.type = VIR_DOMAIN_DEVICE_ADDRESS_TYPE_PCI;
|
||||||
@ -1690,8 +1686,7 @@ qemuDomainAssignDevicePCISlots(virDomainDefPtr def,
|
|||||||
/* This is the first part of the controller, so need
|
/* This is the first part of the controller, so need
|
||||||
* to find a free slot & then reserve this function */
|
* to find a free slot & then reserve this function */
|
||||||
if (qemuDomainPCIAddressReserveNextAddr(addrs, &cont->info,
|
if (qemuDomainPCIAddressReserveNextAddr(addrs, &cont->info,
|
||||||
addr.function,
|
addr.function) < 0) {
|
||||||
false) < 0) {
|
|
||||||
goto error;
|
goto error;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user