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:
Laine Stump 2016-10-19 12:43:04 -04:00
parent 9838cad9cd
commit 9ff9d9f5a9
3 changed files with 43 additions and 70 deletions

View File

@ -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);
} }

View File

@ -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,

View File

@ -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;
} }