summaryrefslogtreecommitdiffstats
path: root/probe_roms.c
diff options
context:
space:
mode:
authorLabun, Marcin <Marcin.Labun@intel.com>2011-03-10 01:41:46 +0100
committerNeilBrown <neilb@suse.de>2011-03-10 01:41:46 +0100
commit3c8bfb5ddd08c929caf9bd6eb36e5c97d1e74dd1 (patch)
tree23586b09a8cfcd5896422fbc17193c9237bd86fb /probe_roms.c
parentimsm : FIX: Assemble dirty array when reshape is in progress (diff)
downloadmdadm-3c8bfb5ddd08c929caf9bd6eb36e5c97d1e74dd1.tar.xz
mdadm-3c8bfb5ddd08c929caf9bd6eb36e5c97d1e74dd1.zip
probe_roms: allow to probe expansion ROMs using vendor and device id.
Adds data offset to PCI expansion ROM Data Structure in resource describing Expansion ROMs. This allows AHCI OROM scanning function to identify AHCI OROM by device id 0x2822 and vendor id 0x8086. Signed-off-by: Marcin Labun <marcin.labun@intel.com> Signed-off-by: NeilBrown <neilb@suse.de>
Diffstat (limited to 'probe_roms.c')
-rw-r--r--probe_roms.c26
1 files changed, 24 insertions, 2 deletions
diff --git a/probe_roms.c b/probe_roms.c
index a8e3a589..6a0b98d8 100644
--- a/probe_roms.c
+++ b/probe_roms.c
@@ -20,6 +20,7 @@
*/
#include "probe_roms.h"
+#include "mdadm.h"
#include <unistd.h>
#include <signal.h>
#include <fcntl.h>
@@ -130,50 +131,60 @@ static void *isa_bus_to_virt(unsigned long addr)
struct resource {
unsigned long start;
unsigned long end;
+ unsigned long data;
const char *name;
};
static struct resource system_rom_resource = {
.name = "System ROM",
.start = 0xf0000,
+ .data = 0,
.end = 0xfffff,
};
static struct resource extension_rom_resource = {
.name = "Extension ROM",
.start = 0xe0000,
+ .data = 0,
.end = 0xeffff,
};
static struct resource adapter_rom_resources[] = { {
.name = "Adapter ROM",
.start = 0xc8000,
+ .data = 0,
.end = 0,
}, {
.name = "Adapter ROM",
.start = 0,
+ .data = 0,
.end = 0,
}, {
.name = "Adapter ROM",
.start = 0,
+ .data = 0,
.end = 0,
}, {
.name = "Adapter ROM",
.start = 0,
+ .data = 0,
.end = 0,
}, {
.name = "Adapter ROM",
.start = 0,
+ .data = 0,
.end = 0,
}, {
.name = "Adapter ROM",
.start = 0,
+ .data = 0,
.end = 0,
} };
static struct resource video_rom_resource = {
.name = "Video ROM",
.start = 0xc0000,
+ .data = 0,
.end = 0xc7fff,
};
@@ -211,7 +222,8 @@ int scan_adapter_roms(scan_fn fn)
if (res->start) {
found = fn(isa_bus_to_virt(res->start),
- isa_bus_to_virt(res->end));
+ isa_bus_to_virt(res->end),
+ isa_bus_to_virt(res->data));
if (found)
break;
} else
@@ -232,6 +244,7 @@ void probe_roms(void)
unsigned long start, length, upper;
unsigned char c;
unsigned int i;
+ __u16 val=0;
if (rom_fd < 0)
return;
@@ -284,14 +297,23 @@ void probe_roms(void)
/* 0 < length <= 0x7f * 512, historically */
length = c * 512;
+ /* Retrieve 16-bit pointer to PCI Data Structure (offset 18h-19h)
+ * The data can be within 64KB forward of the first location
+ * of this code image. The pointer is in little-endian order
+ */
+
+ if (probe_address16(rom + 0x18, &val) != 0)
+ continue;
+ val = __le16_to_cpu(val);
+
/* but accept any length that fits if checksum okay */
if (!length || start + length > upper || !romchecksum(rom, length))
continue;
adapter_rom_resources[i].start = start;
+ adapter_rom_resources[i].data = start + (unsigned long) val;
adapter_rom_resources[i].end = start + length - 1;
start = adapter_rom_resources[i++].end & ~(rom_align - 1);
}
}
-