On Thu, Oct 31, 2013 at 11:10:42AM -0600, Bjorn Helgaas wrote: > On Thu, Oct 31, 2013 at 10:48 AM, Jason Gunthorpe > <jgunthorpe@xxxxxxxxxxxxxxxxxxxx> wrote: > > > Indeed.. I was having problems here because linux was writing 0,0 > > during discovery to the base,limit registers to 'disable' the IO > > window, which triggered a window allocation. So I re-read the PCI > > bridge spec (apparently too quickly) and decided 0,0 was OK, and it > > should be <=. > > > > However, looking again at the spec - it is very clear, < is the > > correct test, and when the values is equal a 4k window should be > > created. > > > > So, I wonder if there is a little bug in the Linux discovery code > > path, should it write FFFF,0 instead? > > Sounds like a bug to me. Do you want to fix it, or point me at it the > place where we write 0,0 so I can fix it? Okay, I remember this now, it was annoying to debug because the PCI driver inits before the serial port starts working, fortunately Sebastian fixed that up, now it is much easier to see what is going on.. The 0,0 write comes from this trace: [<c0111b58>] (mvebu_pcie_handle_iobase_change+0x0/0x140) from [<c0111e70>] (mvebu_pcie_wr_conf+0x1d8/0x318) r5:00000000 r4:c78d0850 [<c0111c98>] (mvebu_pcie_wr_conf+0x0/0x318) from [<c0100e90>] (pci_bus_write_config_word+0x40/0x4c) [<c0100e50>] (pci_bus_write_config_word+0x0/0x4c) from [<c02273d0>] (__pci_bus_size_bridges+0x2e4/0x744) r4:00000000 [<c02270ec>] (__pci_bus_size_bridges+0x0/0x744) from [<c0227344>] (__pci_bus_size_bridges+0x258/0x744) [<c02270ec>] (__pci_bus_size_bridges+0x0/0x744) from [<c0227844>] (pci_bus_size_bridges+0x14/0x18) [<c0227830>] (pci_bus_size_bridges+0x0/0x18) from [<c000cdc0>] (pci_common_init_dev+0x26c/0x2c0) [<c000cb54>] (pci_common_init_dev+0x0/0x2c0) from [<c0111608>] (mvebu_pcie_probe+0x41c/0x480) [<c01111ec>] (mvebu_pcie_probe+0x0/0x480) from [<c01335f0>] (platform_drv_probe+0x1c/0x20) And specifically this code sequence in pci_bridge_check_ranges: pci_read_config_word(bridge, PCI_IO_BASE, &io); if (!io) { pci_write_config_word(bridge, PCI_IO_BASE, 0xf0f0); pci_read_config_word(bridge, PCI_IO_BASE, &io); pci_write_config_word(bridge, PCI_IO_BASE, 0x0); } if (io) b_res[0].flags |= IORESOURCE_IO; If I read this properly, the first sets the IO window to a 4k region at 0xF000, and the second sets the IO window to a 4k region at 0x0. How about this instead: if (!io) { /* Disable the IO window by setting limit < base */ pci_write_config_word(bridge, PCI_IO_BASE, 0x00f0); pci_read_config_word(bridge, PCI_IO_BASE, &io); } /* Bridges without IO support must wire the IO register to 0 */ if (io) b_res[0].flags |= IORESOURCE_IO; 3.2.5.6 is clear that limit < base disables the decoder (however this alone is not reliable if the upper 16 bits are programmed) However, I'm guessing earlier in the sequence the PCI core clears PCI_COMMAND_IO in the bridge command register? If so this isn't a functional problem, it is just weird looking. Which is the reason why my assert triggers, the driver ignores PCI_COMMAND_IO/MEMORY.. Fixing that solves the issue completely. Jason -- To unsubscribe from this list: send the line "unsubscribe linux-pci" in the body of a message to majordomo@xxxxxxxxxxxxxxx More majordomo info at http://vger.kernel.org/majordomo-info.html