Re: [PATCH v15 2/5] vpci/header: emulate PCI_COMMAND register for guests
On 22.05.2024 11:28, Roger Pau Monné wrote: > On Fri, May 17, 2024 at 01:06:12PM -0400, Stewart Hildebrand wrote: >> @@ -754,9 +774,23 @@ static int cf_check init_header(struct pci_dev *pdev) >> return -EOPNOTSUPP; >> } >> >> -/* Setup a handler for the command register. */ >> -rc = vpci_add_register(pdev->vpci, vpci_hw_read16, cmd_write, >> PCI_COMMAND, >> - 2, header); >> +/* >> + * Setup a handler for the command register. >> + * >> + * TODO: If support for emulated bits is added, re-visit how to handle >> + * PCI_COMMAND_PARITY, PCI_COMMAND_SERR, and PCI_COMMAND_FAST_BACK. >> + */ >> +rc = vpci_add_register_mask(pdev->vpci, >> +is_hwdom ? vpci_hw_read16 : guest_cmd_read, >> +cmd_write, PCI_COMMAND, 2, header, 0, 0, >> +PCI_COMMAND_RSVDP_MASK | >> +(is_hwdom ? 0 >> + : PCI_COMMAND_IO | >> +PCI_COMMAND_PARITY | >> +PCI_COMMAND_WAIT | >> +PCI_COMMAND_SERR | >> +PCI_COMMAND_FAST_BACK), > > We want to allow full access to the hw domain and only apply the > PCI_COMMAND_RSVDP_MASK when !is_hwdom in order to keep the current > behavior for dom0. > > I don't think it makes a difference in practice, but we are very lax > in explicitly not applying any of such restrictions to dom0. > > With that fixed: > > Reviewed-by: Roger Pau Monné Makes sense to me, so please feel free to retain my R-b with that adjustment. Jan
Re: [PATCH v15 2/5] vpci/header: emulate PCI_COMMAND register for guests
On Fri, May 17, 2024 at 01:06:12PM -0400, Stewart Hildebrand wrote: > From: Oleksandr Andrushchenko > > Xen and/or Dom0 may have put values in PCI_COMMAND which they expect > to remain unaltered. PCI_COMMAND_SERR bit is a good example: while the > guest's (domU) view of this will want to be zero (for now), the host > having set it to 1 should be preserved, or else we'd effectively be > giving the domU control of the bit. Thus, PCI_COMMAND register needs > proper emulation in order to honor host's settings. > > According to "PCI LOCAL BUS SPECIFICATION, REV. 3.0", section "6.2.2 > Device Control" the reset state of the command register is typically 0, > so when assigning a PCI device use 0 as the initial state for the > guest's (domU) view of the command register. > > Here is the full list of command register bits with notes about > PCI/PCIe specification, and how Xen handles the bit. QEMU's behavior is > also documented here since that is our current reference implementation > for PCI passthrough. > > PCI_COMMAND_IO (bit 0) > PCIe 6.1: RW > PCI LB 3.0: RW > QEMU: (emu_mask) QEMU provides an emulated view of this bit. Guest > writes do not propagate to hardware. QEMU sets this bit to 1 in > hardware if an I/O BAR is exposed to the guest. > Xen domU: (rsvdp_mask) We treat this bit as RsvdP for now since we > don't yet support I/O BARs for domUs. > Xen dom0: We allow dom0 to control this bit freely. > > PCI_COMMAND_MEMORY (bit 1) > PCIe 6.1: RW > PCI LB 3.0: RW > QEMU: (emu_mask) QEMU provides an emulated view of this bit. Guest > writes do not propagate to hardware. QEMU sets this bit to 1 in > hardware if a Memory BAR is exposed to the guest. > Xen domU/dom0: We handle writes to this bit by mapping/unmapping BAR > regions. > Xen domU: For devices assigned to DomUs, memory decoding will be > disabled at the time of initialization. > > PCI_COMMAND_MASTER (bit 2) > PCIe 6.1: RW > PCI LB 3.0: RW > QEMU: Pass through writes to hardware. > Xen domU/dom0: Pass through writes to hardware. > > PCI_COMMAND_SPECIAL (bit 3) > PCIe 6.1: RO, hardwire to 0 > PCI LB 3.0: RW > QEMU: Pass through writes to hardware. > Xen domU/dom0: Pass through writes to hardware. > > PCI_COMMAND_INVALIDATE (bit 4) > PCIe 6.1: RO, hardwire to 0 > PCI LB 3.0: RW > QEMU: Pass through writes to hardware. > Xen domU/dom0: Pass through writes to hardware. > > PCI_COMMAND_VGA_PALETTE (bit 5) > PCIe 6.1: RO, hardwire to 0 > PCI LB 3.0: RW > QEMU: Pass through writes to hardware. > Xen domU/dom0: Pass through writes to hardware. > > PCI_COMMAND_PARITY (bit 6) > PCIe 6.1: RW > PCI LB 3.0: RW > QEMU: (emu_mask) QEMU provides an emulated view of this bit. Guest > writes do not propagate to hardware. > Xen domU: (rsvdp_mask) We treat this bit as RsvdP. > Xen dom0: We allow dom0 to control this bit freely. > > PCI_COMMAND_WAIT (bit 7) > PCIe 6.1: RO, hardwire to 0 > PCI LB 3.0: hardwire to 0 > QEMU: res_mask > Xen domU: (rsvdp_mask) We treat this bit as RsvdP. > Xen dom0: We allow dom0 to control this bit freely. > > PCI_COMMAND_SERR (bit 8) > PCIe 6.1: RW > PCI LB 3.0: RW > QEMU: (emu_mask) QEMU provides an emulated view of this bit. Guest > writes do not propagate to hardware. > Xen domU: (rsvdp_mask) We treat this bit as RsvdP. > Xen dom0: We allow dom0 to control this bit freely. > > PCI_COMMAND_FAST_BACK (bit 9) > PCIe 6.1: RO, hardwire to 0 > PCI LB 3.0: RW > QEMU: (emu_mask) QEMU provides an emulated view of this bit. Guest > writes do not propagate to hardware. > Xen domU: (rsvdp_mask) We treat this bit as RsvdP. > Xen dom0: We allow dom0 to control this bit freely. > > PCI_COMMAND_INTX_DISABLE (bit 10) > PCIe 6.1: RW > PCI LB 3.0: RW > QEMU: (emu_mask) QEMU provides an emulated view of this bit. Guest > writes do not propagate to hardware. QEMU checks if INTx was mapped > for a device. If it is not, then guest can't control > PCI_COMMAND_INTX_DISABLE bit. > Xen domU: We prohibit a guest from enabling INTx if MSI(X) is enabled. > Xen dom0: We allow dom0 to control this bit freely. > > Bits 11-15 > PCIe 6.1: RsvdP > PCI LB 3.0: Reserved > QEMU: res_mask > Xen domU/dom0: rsvdp_mask > > Signed-off-by: Oleksandr Andrushchenko > Signed-off-by: Volodymyr Babchuk > Signed-off-by: Stewart Hildebrand > Reviewed-by: Jan Beulich > --- > RFC: There is an unaddressed question for Roger: should we update the > guest view of the PCI_COMMAND_INTX_DISABLE bit in > msi.c/msix.c:control_write()? See prior discussion at [1]. > In my opinion, I think we should make sure that hardware state and > the guest view are consistent (i.e. don't lie to the guest). > > [1] > https://lore.kernel.org/xen-devel/86b25777-788c-4b9a-8166-a6f8174be...@suse.com/ I think updating the guest view is helpful in case we need to debug issues in the guest. > > In v15: > - add Jan's R-b
[PATCH v15 2/5] vpci/header: emulate PCI_COMMAND register for guests
From: Oleksandr Andrushchenko Xen and/or Dom0 may have put values in PCI_COMMAND which they expect to remain unaltered. PCI_COMMAND_SERR bit is a good example: while the guest's (domU) view of this will want to be zero (for now), the host having set it to 1 should be preserved, or else we'd effectively be giving the domU control of the bit. Thus, PCI_COMMAND register needs proper emulation in order to honor host's settings. According to "PCI LOCAL BUS SPECIFICATION, REV. 3.0", section "6.2.2 Device Control" the reset state of the command register is typically 0, so when assigning a PCI device use 0 as the initial state for the guest's (domU) view of the command register. Here is the full list of command register bits with notes about PCI/PCIe specification, and how Xen handles the bit. QEMU's behavior is also documented here since that is our current reference implementation for PCI passthrough. PCI_COMMAND_IO (bit 0) PCIe 6.1: RW PCI LB 3.0: RW QEMU: (emu_mask) QEMU provides an emulated view of this bit. Guest writes do not propagate to hardware. QEMU sets this bit to 1 in hardware if an I/O BAR is exposed to the guest. Xen domU: (rsvdp_mask) We treat this bit as RsvdP for now since we don't yet support I/O BARs for domUs. Xen dom0: We allow dom0 to control this bit freely. PCI_COMMAND_MEMORY (bit 1) PCIe 6.1: RW PCI LB 3.0: RW QEMU: (emu_mask) QEMU provides an emulated view of this bit. Guest writes do not propagate to hardware. QEMU sets this bit to 1 in hardware if a Memory BAR is exposed to the guest. Xen domU/dom0: We handle writes to this bit by mapping/unmapping BAR regions. Xen domU: For devices assigned to DomUs, memory decoding will be disabled at the time of initialization. PCI_COMMAND_MASTER (bit 2) PCIe 6.1: RW PCI LB 3.0: RW QEMU: Pass through writes to hardware. Xen domU/dom0: Pass through writes to hardware. PCI_COMMAND_SPECIAL (bit 3) PCIe 6.1: RO, hardwire to 0 PCI LB 3.0: RW QEMU: Pass through writes to hardware. Xen domU/dom0: Pass through writes to hardware. PCI_COMMAND_INVALIDATE (bit 4) PCIe 6.1: RO, hardwire to 0 PCI LB 3.0: RW QEMU: Pass through writes to hardware. Xen domU/dom0: Pass through writes to hardware. PCI_COMMAND_VGA_PALETTE (bit 5) PCIe 6.1: RO, hardwire to 0 PCI LB 3.0: RW QEMU: Pass through writes to hardware. Xen domU/dom0: Pass through writes to hardware. PCI_COMMAND_PARITY (bit 6) PCIe 6.1: RW PCI LB 3.0: RW QEMU: (emu_mask) QEMU provides an emulated view of this bit. Guest writes do not propagate to hardware. Xen domU: (rsvdp_mask) We treat this bit as RsvdP. Xen dom0: We allow dom0 to control this bit freely. PCI_COMMAND_WAIT (bit 7) PCIe 6.1: RO, hardwire to 0 PCI LB 3.0: hardwire to 0 QEMU: res_mask Xen domU: (rsvdp_mask) We treat this bit as RsvdP. Xen dom0: We allow dom0 to control this bit freely. PCI_COMMAND_SERR (bit 8) PCIe 6.1: RW PCI LB 3.0: RW QEMU: (emu_mask) QEMU provides an emulated view of this bit. Guest writes do not propagate to hardware. Xen domU: (rsvdp_mask) We treat this bit as RsvdP. Xen dom0: We allow dom0 to control this bit freely. PCI_COMMAND_FAST_BACK (bit 9) PCIe 6.1: RO, hardwire to 0 PCI LB 3.0: RW QEMU: (emu_mask) QEMU provides an emulated view of this bit. Guest writes do not propagate to hardware. Xen domU: (rsvdp_mask) We treat this bit as RsvdP. Xen dom0: We allow dom0 to control this bit freely. PCI_COMMAND_INTX_DISABLE (bit 10) PCIe 6.1: RW PCI LB 3.0: RW QEMU: (emu_mask) QEMU provides an emulated view of this bit. Guest writes do not propagate to hardware. QEMU checks if INTx was mapped for a device. If it is not, then guest can't control PCI_COMMAND_INTX_DISABLE bit. Xen domU: We prohibit a guest from enabling INTx if MSI(X) is enabled. Xen dom0: We allow dom0 to control this bit freely. Bits 11-15 PCIe 6.1: RsvdP PCI LB 3.0: Reserved QEMU: res_mask Xen domU/dom0: rsvdp_mask Signed-off-by: Oleksandr Andrushchenko Signed-off-by: Volodymyr Babchuk Signed-off-by: Stewart Hildebrand Reviewed-by: Jan Beulich --- RFC: There is an unaddressed question for Roger: should we update the guest view of the PCI_COMMAND_INTX_DISABLE bit in msi.c/msix.c:control_write()? See prior discussion at [1]. In my opinion, I think we should make sure that hardware state and the guest view are consistent (i.e. don't lie to the guest). [1] https://lore.kernel.org/xen-devel/86b25777-788c-4b9a-8166-a6f8174be...@suse.com/ In v15: - add Jan's R-b - add blank line after declaration in msi.c:control_write() In v14: - check for 0->1 transition in INTX_DISABLE-setting logic in msi.c:control_write() to match msix.c:control_write() - clear domU-controllable bits in header.c:init_header() In v13: - Update right away (don't defer) PCI_COMMAND_MEMORY bit in guest_cmd variable in cmd_write() - Make comment single line in xen/drivers/vpci/msi.c:control_writ