forked from mit-pdos/xv6-public
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathmp.c
129 lines (116 loc) · 3.14 KB
/
mp.c
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
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
// Multiprocessor support
// Search memory for MP description structures.
// http://developer.intel.com/design/pentium/datashts/24201606.pdf
#include "types.h"
#include "defs.h"
#include "param.h"
#include "memlayout.h"
#include "mp.h"
#include "x86.h"
#include "mmu.h"
#include "proc.h"
struct cpu cpus[NCPU];
int ncpu;
uchar ioapicid;
static uchar sum(uchar *addr, int len) {
int i, sum;
sum = 0;
for (i = 0; i < len; i++)
sum += addr[i];
return sum;
}
// Look for an MP structure in the len bytes at addr.
static struct mp *mpsearch1(uint a, int len) {
uchar *e, *p, *addr;
addr = P2V(a);
e = addr + len;
for (p = addr; p < e; p += sizeof(struct mp))
if (memcmp(p, "_MP_", 4) == 0 && sum(p, sizeof(struct mp)) == 0)
return (struct mp *)p;
return 0;
}
// Search for the MP Floating Pointer Structure, which according to the
// spec is in one of the following three locations:
// 1) in the first KB of the EBDA;
// 2) in the last KB of system base memory;
// 3) in the BIOS ROM between 0xE0000 and 0xFFFFF.
static struct mp *mpsearch(void) {
uchar *bda;
uint p;
struct mp *mp;
bda = (uchar *)P2V(0x400);
if ((p = ((bda[0x0F] << 8) | bda[0x0E]) << 4)) {
if ((mp = mpsearch1(p, 1024)))
return mp;
} else {
p = ((bda[0x14] << 8) | bda[0x13]) * 1024;
if ((mp = mpsearch1(p - 1024, 1024)))
return mp;
}
return mpsearch1(0xF0000, 0x10000);
}
// Search for an MP configuration table. For now,
// don't accept the default configurations (physaddr == 0).
// Check for correct signature, calculate the checksum and,
// if correct, check the version.
// To do: check extended table checksum.
static struct mpconf *mpconfig(struct mp **pmp) {
struct mpconf *conf;
struct mp *mp;
if ((mp = mpsearch()) == 0 || mp->physaddr == 0)
return 0;
conf = (struct mpconf *)P2V((uint)mp->physaddr);
if (memcmp(conf, "PCMP", 4) != 0)
return 0;
if (conf->version != 1 && conf->version != 4)
return 0;
if (sum((uchar *)conf, conf->length) != 0)
return 0;
*pmp = mp;
return conf;
}
void mpinit(void) {
uchar *p, *e;
int ismp;
struct mp *mp;
struct mpconf *conf;
struct mpproc *proc;
struct mpioapic *ioapic;
if ((conf = mpconfig(&mp)) == 0)
panic("Expect to run on an SMP");
ismp = 1;
lapic = (uint *)conf->lapicaddr;
for (p = (uchar *)(conf + 1), e = (uchar *)conf + conf->length; p < e;) {
switch (*p) {
case MPPROC:
proc = (struct mpproc *)p;
if (ncpu < NCPU) {
cpus[ncpu].apicid = proc->apicid; // apicid may differ from ncpu
ncpu++;
}
p += sizeof(struct mpproc);
continue;
case MPIOAPIC:
ioapic = (struct mpioapic *)p;
ioapicid = ioapic->apicno;
p += sizeof(struct mpioapic);
continue;
case MPBUS:
case MPIOINTR:
case MPLINTR:
p += 8;
continue;
default:
ismp = 0;
break;
}
}
if (!ismp)
panic("Didn't find a suitable machine");
if (mp->imcrp) {
// Bochs doesn't support IMCR, so this doesn't run on Bochs.
// But it would on real hardware.
outb(0x22, 0x70); // Select IMCR
outb(0x23, inb(0x23) | 1); // Mask external interrupts.
}
}