2008-12-13 03:32:16 +00:00
|
|
|
|
2011-08-24 11:00:06 +00:00
|
|
|
#define PCI_MAP_REG_START 0x10
|
|
|
|
#define PCI_MAP_ROM_REG 0x30
|
|
|
|
|
|
|
|
#define PCI_MAP_MEMORY 0x00000000
|
|
|
|
#define PCI_MAP_IO 0x00000001
|
|
|
|
|
|
|
|
#define PCI_MAP_MEMORY_TYPE 0x00000007
|
|
|
|
#define PCI_MAP_IO_TYPE 0x00000003
|
|
|
|
|
|
|
|
#define PCI_MAP_MEMORY_TYPE_32BIT 0x00000000
|
|
|
|
#define PCI_MAP_MEMORY_TYPE_32BIT_1M 0x00000002
|
|
|
|
#define PCI_MAP_MEMORY_TYPE_64BIT 0x00000004
|
|
|
|
#define PCI_MAP_MEMORY_TYPE_MASK 0x00000006
|
|
|
|
#define PCI_MAP_MEMORY_CACHABLE 0x00000008
|
|
|
|
#define PCI_MAP_MEMORY_ATTR_MASK 0x0000000e
|
|
|
|
#define PCI_MAP_MEMORY_ADDRESS_MASK 0xfffffff0
|
|
|
|
|
|
|
|
#define PCI_MAP_IO_ATTR_MASK 0x00000003
|
2008-12-13 03:32:16 +00:00
|
|
|
|
|
|
|
u32_t pciGetBaseSize(int bus, int devfn, int index,
|
2010-09-03 09:42:20 +00:00
|
|
|
bool destructive, bool *min)
|
2008-12-13 03:32:16 +00:00
|
|
|
{
|
|
|
|
int offset;
|
|
|
|
u32_t addr1;
|
|
|
|
u32_t addr2;
|
|
|
|
u32_t mask1;
|
|
|
|
u32_t mask2;
|
|
|
|
int bits = 0;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* silently ignore bogus index values. Valid values are 0-6. 0-5 are
|
|
|
|
* the 6 base address registers, and 6 is the ROM base address register.
|
|
|
|
*/
|
|
|
|
if (index < 0 || index > 6)
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
if (min)
|
|
|
|
*min = destructive;
|
|
|
|
|
|
|
|
/* Get the PCI offset */
|
|
|
|
if (index == 6)
|
|
|
|
offset = PCI_MAP_ROM_REG;
|
|
|
|
else
|
|
|
|
offset = PCI_MAP_REG_START + (index << 2);
|
|
|
|
|
|
|
|
addr1 = PciRead32(bus, devfn, offset);
|
|
|
|
/*
|
|
|
|
* Check if this is the second part of a 64 bit address.
|
|
|
|
* XXX need to check how endianness affects 64 bit addresses.
|
|
|
|
*/
|
|
|
|
if (index > 0 && index < 6) {
|
|
|
|
addr2 = PciRead32(bus, devfn, offset - 4);
|
|
|
|
if (PCI_MAP_IS_MEM(addr2) && PCI_MAP_IS64BITMEM(addr2))
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (destructive) {
|
|
|
|
PciWrite32(bus, devfn, offset, 0xffffffff);
|
|
|
|
mask1 = PciRead32(bus, devfn, offset);
|
|
|
|
PciWrite32(bus, devfn, offset, addr1);
|
|
|
|
} else {
|
|
|
|
mask1 = addr1;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Check if this is the first part of a 64 bit address. */
|
|
|
|
if (index < 5 && PCI_MAP_IS_MEM(mask1) && PCI_MAP_IS64BITMEM(mask1))
|
|
|
|
{
|
|
|
|
if (PCIGETMEMORY(mask1) == 0)
|
|
|
|
{
|
|
|
|
addr2 = PciRead32(bus, devfn, offset + 4);
|
|
|
|
if (destructive)
|
|
|
|
{
|
|
|
|
PciWrite32(bus, devfn, offset + 4, 0xffffffff);
|
|
|
|
mask2 = PciRead32(bus, devfn, offset + 4);
|
|
|
|
PciWrite32(bus, devfn, offset + 4, addr2);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
mask2 = addr2;
|
|
|
|
}
|
|
|
|
if (mask2 == 0)
|
|
|
|
return 0;
|
|
|
|
bits = 32;
|
|
|
|
while ((mask2 & 1) == 0)
|
|
|
|
{
|
|
|
|
bits++;
|
|
|
|
mask2 >>= 1;
|
|
|
|
}
|
|
|
|
if (bits > 32)
|
|
|
|
return bits;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (index < 6)
|
|
|
|
if (PCI_MAP_IS_MEM(mask1))
|
|
|
|
mask1 = PCIGETMEMORY(mask1);
|
|
|
|
else
|
|
|
|
mask1 = PCIGETIO(mask1);
|
|
|
|
else
|
|
|
|
mask1 = PCIGETROM(mask1);
|
|
|
|
if (mask1 == 0)
|
|
|
|
return 0;
|
|
|
|
bits = 0;
|
|
|
|
while ((mask1 & 1) == 0) {
|
|
|
|
bits++;
|
|
|
|
mask1 >>= 1;
|
|
|
|
}
|
|
|
|
/* I/O maps can be no larger than 8 bits */
|
|
|
|
|
|
|
|
if ((index < 6) && PCI_MAP_IS_IO(addr1) && bits > 8)
|
|
|
|
bits = 8;
|
|
|
|
/* ROM maps can be no larger than 24 bits */
|
|
|
|
if (index == 6 && bits > 24)
|
|
|
|
bits = 24;
|
|
|
|
return bits;
|
|
|
|
}
|