From: mikem@beardog.cca.cpqcorp.net This patch eliminates the bad assumption that all of our PCI BARs will always be 32-bits. Tested against the 2.6.2 kernel. This is required to support the Pinnacles architecture. It is already in the 2.4 tree. --- drivers/block/cciss.c | 74 +++++++++++++++++++++++++++++++++++++------------- drivers/block/cciss.h | 4 +- 2 files changed, 57 insertions(+), 21 deletions(-) diff -puN drivers/block/cciss.c~cciss-01-pci-bar-fix drivers/block/cciss.c --- 25/drivers/block/cciss.c~cciss-01-pci-bar-fix 2004-02-04 20:20:54.000000000 -0800 +++ 25-akpm/drivers/block/cciss.c 2004-02-04 20:20:54.000000000 -0800 @@ -2078,16 +2078,51 @@ static void release_io_mem(ctlr_info_t * c->io_mem_addr = 0; c->io_mem_length = 0; } + +static int find_PCI_BAR_index(struct pci_dev *pdev, + unsigned long pci_bar_addr) +{ + int i, offset, mem_type, bar_type; + if (pci_bar_addr == PCI_BASE_ADDRESS_0) /* looking for BAR zero? */ + return 0; + offset = 0; + for (i=0; idevice; irq = pdev->irq; - for(i=0; i<6; i++) - addr[i] = pdev->resource[i].start; - (void) pci_read_config_word(pdev, PCI_COMMAND,&command); (void) pci_read_config_byte(pdev, PCI_CLASS_REVISION, &revision); (void) pci_read_config_byte(pdev, PCI_CACHE_LINE_SIZE, @@ -2125,14 +2157,13 @@ static int cciss_pci_init(ctlr_info_t *c } /* search for our IO range so we can protect it */ - for(i=0; i<6; i++) + for(i=0; iresource[i].flags & 0x01 ) - { - c->io_mem_addr = pdev->resource[i].start; - c->io_mem_length = pdev->resource[i].end - - pdev->resource[i].start +1; + if( pdev_resource_flags(pdev, i) & 0x01 ) { + c->io_mem_addr = pdev_resource_start(pdev, i); + c->io_mem_length = pdev_resource_end(pdev, i) - + pdev_resource_start(pdev, i) +1; #ifdef CCISS_DEBUG printk("IO value found base_addr[%d] %lx %lx\n", i, c->io_mem_addr, c->io_mem_length); @@ -2155,7 +2186,7 @@ static int cciss_pci_init(ctlr_info_t *c printk("device_id = %x\n", device_id); printk("command = %x\n", command); for(i=0; i<6; i++) - printk("addr[%d] = %x\n", i, addr[i]); + printk("addr[%d] = %x\n", i, pci_resource_start(pdev, i); printk("revision = %x\n", revision); printk("irq = %x\n", irq); printk("cache_line_size = %x\n", cache_line_size); @@ -2170,7 +2201,7 @@ static int cciss_pci_init(ctlr_info_t *c * table */ - c->paddr = addr[0] ; /* addressing mode bits already removed */ + c->paddr = pci_resource_start(pdev, 0); /* addressing mode bits already removed */ #ifdef CCISS_DEBUG printk("address 0 = %x\n", c->paddr); #endif /* CCISS_DEBUG */ @@ -2192,22 +2223,27 @@ static int cciss_pci_init(ctlr_info_t *c /* get the address index number */ cfg_base_addr = readl(c->vaddr + SA5_CTCFG_OFFSET); - /* I am not prepared to deal with a 64 bit address value */ - cfg_base_addr &= 0xffff; + cfg_base_addr &= (__u32) 0x0000ffff; #ifdef CCISS_DEBUG printk("cfg base address = %x\n", cfg_base_addr); #endif /* CCISS_DEBUG */ - cfg_base_addr_index = (cfg_base_addr - PCI_BASE_ADDRESS_0)/4; + cfg_base_addr_index = + find_PCI_BAR_index(pdev, cfg_base_addr); #ifdef CCISS_DEBUG printk("cfg base address index = %x\n", cfg_base_addr_index); #endif /* CCISS_DEBUG */ + if (cfg_base_addr_index == -1) { + printk(KERN_WARNING "cciss: Cannot find cfg_base_addr_index\n"); + release_io_mem(hba[i]); + return -1; + } cfg_offset = readl(c->vaddr + SA5_CTMEM_OFFSET); #ifdef CCISS_DEBUG printk("cfg offset = %x\n", cfg_offset); #endif /* CCISS_DEBUG */ c->cfgtable = (CfgTable_struct *) - remap_pci_mem((addr[cfg_base_addr_index] & 0xfffffff0) + remap_pci_mem(pci_resource_start(pdev, cfg_base_addr_index) + cfg_offset, sizeof(CfgTable_struct)); c->board_id = board_id; diff -puN drivers/block/cciss.h~cciss-01-pci-bar-fix drivers/block/cciss.h --- 25/drivers/block/cciss.h~cciss-01-pci-bar-fix 2004-02-04 20:20:54.000000000 -0800 +++ 25-akpm/drivers/block/cciss.h 2004-02-04 20:20:54.000000000 -0800 @@ -42,8 +42,8 @@ struct ctlr_info char firm_ver[4]; // Firmware version struct pci_dev *pdev; __u32 board_id; - ulong vaddr; - __u32 paddr; + unsigned long vaddr; + unsigned long paddr; unsigned long io_mem_addr; unsigned long io_mem_length; CfgTable_struct *cfgtable; _