2006-06-21 03:53:07 +02:00
|
|
|
#include "types.h"
|
|
|
|
#include "mp.h"
|
|
|
|
#include "defs.h"
|
|
|
|
#include "memlayout.h"
|
2006-06-22 03:28:57 +02:00
|
|
|
#include "param.h"
|
|
|
|
#include "x86.h"
|
2006-06-28 18:35:03 +02:00
|
|
|
#include "traps.h"
|
2006-06-22 03:28:57 +02:00
|
|
|
#include "mmu.h"
|
2006-06-21 03:53:07 +02:00
|
|
|
|
2006-06-22 03:28:57 +02:00
|
|
|
/*
|
|
|
|
* Credit: Plan 9 sources, Intel MP spec, and Cliff Frey
|
|
|
|
*/
|
|
|
|
|
|
|
|
enum { /* Local APIC registers */
|
|
|
|
LAPIC_ID = 0x0020, /* ID */
|
|
|
|
LAPIC_VER = 0x0030, /* Version */
|
|
|
|
LAPIC_TPR = 0x0080, /* Task Priority */
|
|
|
|
LAPIC_APR = 0x0090, /* Arbitration Priority */
|
|
|
|
LAPIC_PPR = 0x00A0, /* Processor Priority */
|
|
|
|
LAPIC_EOI = 0x00B0, /* EOI */
|
|
|
|
LAPIC_LDR = 0x00D0, /* Logical Destination */
|
|
|
|
LAPIC_DFR = 0x00E0, /* Destination Format */
|
|
|
|
LAPIC_SVR = 0x00F0, /* Spurious Interrupt Vector */
|
|
|
|
LAPIC_ISR = 0x0100, /* Interrupt Status (8 registers) */
|
|
|
|
LAPIC_TMR = 0x0180, /* Trigger Mode (8 registers) */
|
|
|
|
LAPIC_IRR = 0x0200, /* Interrupt Request (8 registers) */
|
|
|
|
LAPIC_ESR = 0x0280, /* Error Status */
|
|
|
|
LAPIC_ICRLO = 0x0300, /* Interrupt Command */
|
|
|
|
LAPIC_ICRHI = 0x0310, /* Interrupt Command [63:32] */
|
|
|
|
LAPIC_TIMER = 0x0320, /* Local Vector Table 0 (TIMER) */
|
|
|
|
LAPIC_PCINT = 0x0340, /* Performance Counter LVT */
|
|
|
|
LAPIC_LINT0 = 0x0350, /* Local Vector Table 1 (LINT0) */
|
|
|
|
LAPIC_LINT1 = 0x0360, /* Local Vector Table 2 (LINT1) */
|
|
|
|
LAPIC_ERROR = 0x0370, /* Local Vector Table 3 (ERROR) */
|
|
|
|
LAPIC_TICR = 0x0380, /* Timer Initial Count */
|
|
|
|
LAPIC_TCCR = 0x0390, /* Timer Current Count */
|
|
|
|
LAPIC_TDCR = 0x03E0, /* Timer Divide Configuration */
|
|
|
|
};
|
|
|
|
|
|
|
|
enum { /* LAPIC_SVR */
|
|
|
|
LAPIC_ENABLE = 0x00000100, /* Unit Enable */
|
|
|
|
LAPIC_FOCUS = 0x00000200, /* Focus Processor Checking Disable */
|
|
|
|
};
|
|
|
|
|
|
|
|
enum { /* LAPIC_ICRLO */
|
|
|
|
/* [14] IPI Trigger Mode Level (RW) */
|
|
|
|
LAPIC_DEASSERT = 0x00000000, /* Deassert level-sensitive interrupt */
|
|
|
|
LAPIC_ASSERT = 0x00004000, /* Assert level-sensitive interrupt */
|
|
|
|
|
|
|
|
/* [17:16] Remote Read Status */
|
|
|
|
LAPIC_INVALID = 0x00000000, /* Invalid */
|
|
|
|
LAPIC_WAIT = 0x00010000, /* In-Progress */
|
|
|
|
LAPIC_VALID = 0x00020000, /* Valid */
|
|
|
|
|
|
|
|
/* [19:18] Destination Shorthand */
|
|
|
|
LAPIC_FIELD = 0x00000000, /* No shorthand */
|
|
|
|
LAPIC_SELF = 0x00040000, /* Self is single destination */
|
|
|
|
LAPIC_ALLINC = 0x00080000, /* All including self */
|
|
|
|
LAPIC_ALLEXC = 0x000C0000, /* All Excluding self */
|
|
|
|
};
|
|
|
|
|
|
|
|
enum { /* LAPIC_ESR */
|
|
|
|
LAPIC_SENDCS = 0x00000001, /* Send CS Error */
|
|
|
|
LAPIC_RCVCS = 0x00000002, /* Receive CS Error */
|
|
|
|
LAPIC_SENDACCEPT = 0x00000004, /* Send Accept Error */
|
|
|
|
LAPIC_RCVACCEPT = 0x00000008, /* Receive Accept Error */
|
|
|
|
LAPIC_SENDVECTOR = 0x00000020, /* Send Illegal Vector */
|
|
|
|
LAPIC_RCVVECTOR = 0x00000040, /* Receive Illegal Vector */
|
|
|
|
LAPIC_REGISTER = 0x00000080, /* Illegal Register Address */
|
|
|
|
};
|
|
|
|
|
|
|
|
enum { /* LAPIC_TIMER */
|
|
|
|
/* [17] Timer Mode (RW) */
|
|
|
|
LAPIC_ONESHOT = 0x00000000, /* One-shot */
|
|
|
|
LAPIC_PERIODIC = 0x00020000, /* Periodic */
|
|
|
|
|
|
|
|
/* [19:18] Timer Base (RW) */
|
|
|
|
LAPIC_CLKIN = 0x00000000, /* use CLKIN as input */
|
|
|
|
LAPIC_TMBASE = 0x00040000, /* use TMBASE */
|
|
|
|
LAPIC_DIVIDER = 0x00080000, /* use output of the divider */
|
|
|
|
};
|
|
|
|
|
|
|
|
enum { /* LAPIC_TDCR */
|
|
|
|
LAPIC_X2 = 0x00000000, /* divide by 2 */
|
|
|
|
LAPIC_X4 = 0x00000001, /* divide by 4 */
|
|
|
|
LAPIC_X8 = 0x00000002, /* divide by 8 */
|
|
|
|
LAPIC_X16 = 0x00000003, /* divide by 16 */
|
|
|
|
LAPIC_X32 = 0x00000008, /* divide by 32 */
|
|
|
|
LAPIC_X64 = 0x00000009, /* divide by 64 */
|
|
|
|
LAPIC_X128 = 0x0000000A, /* divide by 128 */
|
|
|
|
LAPIC_X1 = 0x0000000B, /* divide by 1 */
|
|
|
|
};
|
|
|
|
|
|
|
|
#define APBOOTCODE 0x7000 // XXX hack
|
2006-06-26 18:40:43 +02:00
|
|
|
#define MPSTACK 512
|
2006-06-22 03:28:57 +02:00
|
|
|
|
2006-06-26 18:40:43 +02:00
|
|
|
static struct MP* mp; // The MP floating point structure
|
2006-06-22 03:28:57 +02:00
|
|
|
static uint32_t *lapicaddr;
|
|
|
|
static struct cpu {
|
2006-06-26 18:40:43 +02:00
|
|
|
uint8_t apicid; // Local APIC ID
|
|
|
|
int lintr[2]; // Local APIC
|
|
|
|
char mpstack[MPSTACK]; // per-cpu start-up stack, only used to get into main()
|
2006-06-22 22:47:23 +02:00
|
|
|
} cpus[NCPU];
|
2006-06-21 03:53:07 +02:00
|
|
|
static int ncpu;
|
2006-06-22 03:28:57 +02:00
|
|
|
static struct cpu *bcpu;
|
|
|
|
|
|
|
|
static int
|
|
|
|
lapic_read(int r)
|
|
|
|
{
|
|
|
|
return *(lapicaddr+(r/sizeof(*lapicaddr)));
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
lapic_write(int r, int data)
|
|
|
|
{
|
|
|
|
*(lapicaddr+(r/sizeof(*lapicaddr))) = data;
|
|
|
|
}
|
|
|
|
|
2006-06-28 18:35:03 +02:00
|
|
|
|
|
|
|
void
|
|
|
|
lapic_timerinit()
|
|
|
|
{
|
|
|
|
cprintf("%d: init timer\n", cpu());
|
|
|
|
lapic_write(LAPIC_TDCR, LAPIC_X1);
|
|
|
|
lapic_write(LAPIC_TIMER, LAPIC_CLKIN | LAPIC_PERIODIC | (IRQ_OFFSET + IRQ_TIMER));
|
|
|
|
lapic_write(LAPIC_TCCR, 1000000);
|
|
|
|
lapic_write(LAPIC_TICR, 1000000);
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
lapic_timerintr()
|
|
|
|
{
|
|
|
|
cprintf("%d: timer interrupt!\n", cpu());
|
|
|
|
lapic_write (LAPIC_EOI, 0);
|
|
|
|
}
|
|
|
|
|
2006-06-22 22:47:23 +02:00
|
|
|
void
|
2006-06-22 03:28:57 +02:00
|
|
|
lapic_init(int c)
|
|
|
|
{
|
|
|
|
uint32_t r, lvt;
|
|
|
|
|
|
|
|
cprintf("lapic_init %d\n", c);
|
2006-06-28 18:35:03 +02:00
|
|
|
|
|
|
|
irq_setmask_8259A(0xFFFF);
|
|
|
|
|
2006-06-22 03:28:57 +02:00
|
|
|
lapic_write(LAPIC_DFR, 0xFFFFFFFF);
|
|
|
|
r = (lapic_read(LAPIC_ID)>>24) & 0xFF;
|
|
|
|
lapic_write(LAPIC_LDR, (1<<r)<<24);
|
|
|
|
lapic_write(LAPIC_TPR, 0xFF);
|
|
|
|
lapic_write(LAPIC_SVR, LAPIC_ENABLE|(IRQ_OFFSET+IRQ_SPURIOUS));
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Set the local interrupts. It's likely these should just be
|
|
|
|
* masked off for SMP mode as some Pentium Pros have problems if
|
|
|
|
* LINT[01] are set to ExtINT.
|
|
|
|
* Acknowledge any outstanding interrupts.
|
|
|
|
*/
|
2006-06-22 22:47:23 +02:00
|
|
|
lapic_write(LAPIC_LINT0, cpus[c].lintr[0]);
|
|
|
|
lapic_write(LAPIC_LINT1, cpus[c].lintr[1]);
|
2006-06-22 03:28:57 +02:00
|
|
|
lapic_write(LAPIC_EOI, 0);
|
|
|
|
|
|
|
|
lvt = (lapic_read(LAPIC_VER)>>16) & 0xFF;
|
|
|
|
if(lvt >= 4)
|
|
|
|
lapic_write(LAPIC_PCINT, APIC_IMASK);
|
|
|
|
lapic_write(LAPIC_ERROR, IRQ_OFFSET+IRQ_ERROR);
|
|
|
|
lapic_write(LAPIC_ESR, 0);
|
|
|
|
lapic_read(LAPIC_ESR);
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Issue an INIT Level De-Assert to synchronise arbitration ID's.
|
|
|
|
*/
|
|
|
|
lapic_write(LAPIC_ICRHI, 0);
|
|
|
|
lapic_write(LAPIC_ICRLO, LAPIC_ALLINC|APIC_LEVEL|LAPIC_DEASSERT|APIC_INIT);
|
|
|
|
while(lapic_read(LAPIC_ICRLO) & APIC_DELIVS)
|
|
|
|
;
|
2006-06-21 03:53:07 +02:00
|
|
|
|
2006-06-22 03:28:57 +02:00
|
|
|
cprintf("Done init of an apic\n");
|
|
|
|
}
|
|
|
|
|
2006-06-28 18:35:03 +02:00
|
|
|
void
|
|
|
|
lapic_enableintr(void)
|
2006-06-22 03:28:57 +02:00
|
|
|
{
|
|
|
|
lapic_write(LAPIC_TPR, 0);
|
|
|
|
}
|
|
|
|
|
2006-06-28 18:44:41 +02:00
|
|
|
void
|
|
|
|
lapic_disableintr(void)
|
|
|
|
{
|
|
|
|
lapic_write(LAPIC_TPR, 0xFF);
|
|
|
|
}
|
|
|
|
|
2006-06-22 03:28:57 +02:00
|
|
|
int
|
2006-06-22 22:47:23 +02:00
|
|
|
cpu(void)
|
2006-06-22 03:28:57 +02:00
|
|
|
{
|
|
|
|
return (lapic_read(LAPIC_ID)>>24) & 0xFF;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
lapic_startap(struct cpu *c, int v)
|
|
|
|
{
|
|
|
|
int crhi, i;
|
|
|
|
volatile int j = 0;
|
|
|
|
|
|
|
|
crhi = c->apicid<<24;
|
|
|
|
lapic_write(LAPIC_ICRHI, crhi);
|
|
|
|
lapic_write(LAPIC_ICRLO, LAPIC_FIELD|APIC_LEVEL|LAPIC_ASSERT|APIC_INIT);
|
|
|
|
|
|
|
|
while (j++ < 10000) {;}
|
|
|
|
lapic_write(LAPIC_ICRLO, LAPIC_FIELD|APIC_LEVEL|LAPIC_DEASSERT|APIC_INIT);
|
|
|
|
|
|
|
|
while (j++ < 1000000) {;}
|
|
|
|
|
|
|
|
// in p9 code, this was i < 2, which is what the spec says on page B-3
|
|
|
|
for(i = 0; i < 1; i++){
|
|
|
|
lapic_write(LAPIC_ICRHI, crhi);
|
|
|
|
lapic_write(LAPIC_ICRLO, LAPIC_FIELD|APIC_EDGE|APIC_STARTUP|(v/PGSIZE));
|
|
|
|
while (j++ < 100000) {;}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static struct MP*
|
2006-06-21 03:53:07 +02:00
|
|
|
mp_scan(uint8_t *addr, int len)
|
|
|
|
{
|
|
|
|
uint8_t *e, *p, sum;
|
|
|
|
int i;
|
|
|
|
|
|
|
|
cprintf("scanning: 0x%x\n", (uint32_t)addr);
|
|
|
|
e = addr+len;
|
2006-06-22 03:28:57 +02:00
|
|
|
for(p = addr; p < e; p += sizeof(struct MP)){
|
2006-06-21 03:53:07 +02:00
|
|
|
if(memcmp(p, "_MP_", 4))
|
|
|
|
continue;
|
|
|
|
sum = 0;
|
2006-06-22 03:28:57 +02:00
|
|
|
for(i = 0; i < sizeof(struct MP); i++)
|
2006-06-21 03:53:07 +02:00
|
|
|
sum += p[i];
|
|
|
|
if(sum == 0)
|
2006-06-22 03:28:57 +02:00
|
|
|
return (struct MP *)p;
|
2006-06-21 03:53:07 +02:00
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2006-06-22 03:28:57 +02:00
|
|
|
static struct MP*
|
2006-06-21 03:53:07 +02:00
|
|
|
mp_search(void)
|
|
|
|
{
|
|
|
|
uint8_t *bda;
|
|
|
|
uint32_t p;
|
2006-06-22 03:28:57 +02:00
|
|
|
struct MP *mp;
|
2006-06-21 03:53:07 +02:00
|
|
|
|
|
|
|
/*
|
|
|
|
* 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.
|
|
|
|
*/
|
2006-06-25 00:47:06 +02:00
|
|
|
bda = (uint8_t*) 0x400;
|
2006-06-21 03:53:07 +02:00
|
|
|
if((p = (bda[0x0F]<<8)|bda[0x0E])){
|
2006-06-25 00:47:06 +02:00
|
|
|
if((mp = mp_scan((uint8_t*) p, 1024)))
|
2006-06-21 03:53:07 +02:00
|
|
|
return mp;
|
|
|
|
}
|
|
|
|
else{
|
|
|
|
p = ((bda[0x14]<<8)|bda[0x13])*1024;
|
2006-06-25 00:47:06 +02:00
|
|
|
if((mp = mp_scan((uint8_t*)p-1024, 1024)))
|
2006-06-21 03:53:07 +02:00
|
|
|
return mp;
|
|
|
|
}
|
2006-06-25 00:47:06 +02:00
|
|
|
return mp_scan((uint8_t*)0xF0000, 0x10000);
|
2006-06-21 03:53:07 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
static int
|
|
|
|
mp_detect(void)
|
|
|
|
{
|
2006-06-22 03:28:57 +02:00
|
|
|
struct MPCTB *pcmp;
|
2006-06-21 03:53:07 +02:00
|
|
|
uint8_t *p, sum;
|
|
|
|
uint32_t length;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* 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.
|
|
|
|
*/
|
2006-06-22 03:28:57 +02:00
|
|
|
if((mp = mp_search()) == 0 || mp->physaddr == 0)
|
2006-06-21 03:53:07 +02:00
|
|
|
return 1;
|
|
|
|
|
2006-06-25 00:47:06 +02:00
|
|
|
pcmp = (struct MPCTB *) mp->physaddr;
|
2006-06-21 03:53:07 +02:00
|
|
|
if(memcmp(pcmp, "PCMP", 4))
|
|
|
|
return 2;
|
|
|
|
|
|
|
|
length = pcmp->length;
|
|
|
|
sum = 0;
|
|
|
|
for(p = (uint8_t*)pcmp; length; length--)
|
|
|
|
sum += *p++;
|
|
|
|
|
|
|
|
if(sum || (pcmp->version != 1 && pcmp->version != 4))
|
|
|
|
return 3;
|
|
|
|
|
2006-06-28 18:35:03 +02:00
|
|
|
cprintf("Mp spec rev #: %x\n", mp->specrev);
|
2006-06-21 03:53:07 +02:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2006-06-22 03:28:57 +02:00
|
|
|
int
|
|
|
|
mp_isbcpu()
|
|
|
|
{
|
|
|
|
if (bcpu == 0) return 1;
|
|
|
|
else return 0;
|
|
|
|
}
|
|
|
|
|
2006-06-21 03:53:07 +02:00
|
|
|
void
|
2006-06-22 03:28:57 +02:00
|
|
|
mp_init()
|
2006-06-21 03:53:07 +02:00
|
|
|
{
|
|
|
|
int r;
|
|
|
|
uint8_t *p, *e;
|
2006-06-22 03:28:57 +02:00
|
|
|
struct MPCTB *mpctb;
|
|
|
|
struct MPPE *proc;
|
2006-06-25 00:47:06 +02:00
|
|
|
int c;
|
|
|
|
extern int main();
|
2006-06-21 03:53:07 +02:00
|
|
|
|
|
|
|
ncpu = 0;
|
|
|
|
if ((r = mp_detect()) != 0) return;
|
2006-06-22 03:28:57 +02:00
|
|
|
|
|
|
|
cprintf ("This computer is a multiprocessor!\n");
|
2006-06-21 03:53:07 +02:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Run through the table saving information needed for starting
|
|
|
|
* application processors and initialising any I/O APICs. The table
|
|
|
|
* is guaranteed to be in order such that only one pass is necessary.
|
|
|
|
*/
|
2006-06-25 00:47:06 +02:00
|
|
|
mpctb = (struct MPCTB *) mp->physaddr;
|
|
|
|
lapicaddr = (uint32_t *) mpctb->lapicaddr;
|
2006-06-22 03:28:57 +02:00
|
|
|
cprintf("apicaddr: %x\n", lapicaddr);
|
|
|
|
p = ((uint8_t*)mpctb)+sizeof(struct MPCTB);
|
|
|
|
e = ((uint8_t*)mpctb)+mpctb->length;
|
2006-06-21 03:53:07 +02:00
|
|
|
|
|
|
|
while(p < e) {
|
|
|
|
switch(*p){
|
2006-06-22 03:28:57 +02:00
|
|
|
case MPPROCESSOR:
|
|
|
|
proc = (struct MPPE *) p;
|
2006-06-22 22:47:23 +02:00
|
|
|
cpus[ncpu].apicid = proc->apicid;
|
|
|
|
cpus[ncpu].lintr[0] = APIC_IMASK;
|
|
|
|
cpus[ncpu].lintr[1] = APIC_IMASK;
|
|
|
|
cprintf("a processor %x\n", cpus[ncpu].apicid);
|
2006-06-22 03:28:57 +02:00
|
|
|
if (proc->flags & MPBP) {
|
2006-06-22 22:47:23 +02:00
|
|
|
bcpu = &cpus[ncpu];
|
2006-06-22 03:28:57 +02:00
|
|
|
}
|
2006-06-21 03:53:07 +02:00
|
|
|
ncpu++;
|
2006-06-22 03:28:57 +02:00
|
|
|
p += sizeof(struct MPPE);
|
2006-06-21 03:53:07 +02:00
|
|
|
continue;
|
2006-06-22 03:28:57 +02:00
|
|
|
case MPBUS:
|
|
|
|
p += sizeof(struct MPBE);
|
2006-06-21 03:53:07 +02:00
|
|
|
continue;
|
2006-06-22 03:28:57 +02:00
|
|
|
case MPIOAPIC:
|
|
|
|
cprintf("an I/O APIC\n");
|
|
|
|
p += sizeof(struct MPIOAPIC);
|
2006-06-21 03:53:07 +02:00
|
|
|
continue;
|
2006-06-22 03:28:57 +02:00
|
|
|
case MPIOINTR:
|
|
|
|
p += sizeof(struct MPIE);
|
2006-06-21 03:53:07 +02:00
|
|
|
continue;
|
|
|
|
default:
|
|
|
|
cprintf("mpinit: unknown PCMP type 0x%x (e-p 0x%x)\n", *p, e-p);
|
|
|
|
while(p < e){
|
|
|
|
cprintf("%uX ", *p);
|
|
|
|
p++;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2006-06-22 03:28:57 +02:00
|
|
|
|
2006-06-22 22:47:23 +02:00
|
|
|
lapic_init(bcpu-cpus);
|
|
|
|
cprintf("ncpu: %d boot %d\n", ncpu, bcpu-cpus);
|
2006-06-21 03:53:07 +02:00
|
|
|
|
2006-06-22 03:28:57 +02:00
|
|
|
extern uint8_t _binary_bootother_start[], _binary_bootother_size[];
|
2006-06-25 00:47:06 +02:00
|
|
|
memmove((void *) APBOOTCODE,_binary_bootother_start,
|
2006-06-22 03:28:57 +02:00
|
|
|
(uint32_t) _binary_bootother_size);
|
|
|
|
|
|
|
|
acquire_spinlock(&kernel_lock);
|
2006-06-25 00:47:06 +02:00
|
|
|
for(c = 0; c < ncpu; c++){
|
|
|
|
if (cpus+c == bcpu) continue;
|
|
|
|
cprintf ("starting processor %d\n", c);
|
|
|
|
release_grant_spinlock(&kernel_lock, c);
|
2006-06-26 18:40:43 +02:00
|
|
|
*(unsigned *)(APBOOTCODE-4) = (unsigned) (cpus[c].mpstack) + MPSTACK; // tell it what to use for %esp
|
2006-06-25 00:47:06 +02:00
|
|
|
*(unsigned *)(APBOOTCODE-8) = (unsigned)&main; // tell it where to jump to
|
|
|
|
lapic_startap(cpus + c, (uint32_t) APBOOTCODE);
|
2006-06-22 03:28:57 +02:00
|
|
|
acquire_spinlock(&kernel_lock);
|
2006-06-25 00:47:06 +02:00
|
|
|
cprintf ("done starting processor %d\n", c);
|
2006-06-22 03:28:57 +02:00
|
|
|
}
|
2006-06-21 03:53:07 +02:00
|
|
|
}
|