summaryrefslogtreecommitdiff
path: root/xc/programs/Xserver/hw/xfree86/int10/pci.c
blob: 9b523ae82c5466faefb6f3d9d2963e02f9aa804e (plain)
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;
}