1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
|
/* $XFree86: xc/programs/Xserver/hw/xfree86/int10/pci.c,v 1.6 2001/01/06 20:19:13 tsi Exp $ */
/*
* XFree86 int10 module
* execute BIOS int 10h calls in x86 real mode environment
* Copyright 1999 Egbert Eich
*/
#include "xf86Pci.h"
#include "xf86.h"
#include "xf86_ansic.h"
#define _INT10_PRIVATE
#include "xf86int10.h"
int
mapPciRom(xf86Int10InfoPtr pInt, unsigned char * address)
{
PCITAG tag;
unsigned long offset = 0;
unsigned char *mem, *ptr;
int length;
pciVideoPtr pvp = xf86GetPciInfoForEntity(pInt->entityIndex);
if (pvp == NULL) {
#ifdef DEBUG
ErrorF("mapPciRom: no PCI info\n");
#endif
return 0;
}
tag = pciTag(pvp->bus,pvp->device,pvp->func);
length = 1 << pvp->biosSize;
/* Read in entire PCI ROM */
mem = ptr = xnfcalloc(length, 1);
if (xf86ReadPciBIOS(offset, tag, -1, ptr, length) < length) {
xfree(mem);
#ifdef DEBUG
ErrorF("mapPciRom: cannot read BIOS\n");
#endif
return 0;
}
while ((ptr[0] == 0x55) && (ptr[1] == 0xAA)) {
unsigned short data_off = ptr[0x18] | (ptr[0x19] << 8);
unsigned char *data = ptr + data_off;
unsigned char type;
if ((data[0] != 'P') ||
(data[1] != 'C') ||
(data[2] != 'I') ||
(data[3] != 'R'))
break;
type = data[0x14];
#ifdef PRINT_PCI
ErrorF("data segment in BIOS: 0x%x, type: 0x%x\n", data_off, type);
#endif
if (type) { /* not PC-AT image: find next one */
unsigned int image_length;
unsigned char indicator = data[0x15];
if (indicator & 0x80) /* last image */
break;
image_length = (data[0x10] | (data[0x11] << 8)) << 9;
#ifdef PRINT_PCI
ErrorF("data image length: 0x%x, ind: 0x%x\n",
image_length, indicator);
#endif
ptr += image_length;
continue;
}
/* OK, we have a PC Image */
length = ptr[2] << 9;
#ifdef PRINT_PCI
ErrorF("BIOS length: 0x%x\n", length);
#endif
break;
}
if (length > 0)
memcpy(address, ptr, length);
/* unmap/close/disable PCI bios mem */
xfree(mem);
#ifdef DEBUG
if (!length)
ErrorF("mapPciRom: no BIOS found\n");
#ifdef PRINT_PCI
else
dprint(address,0x20);
#endif
#endif
return length;
}
|