# xv6 啟動流程
:::success
**File:** main.c
:::
- BIOS -> boot section -> main -> scheduler 的詳細流程在 [ch1](https://hackmd.io/s/BJjmDiZmX#)、[ch5](https://hackmd.io/s/ry0BqIeH7#)及[Appendix B](https://hackmd.io/s/ByDvT3BQm),本文強調 CPU0 以外的 CPU 啟動流程及更詳細的 main 解析。
## Code: startothers
- 在 main 初始化一些設備後,會先呼叫 startothers,再呼叫 mpmain 來完成 cpu 的設定及呼叫 scheduler。
```c=68
static void
startothers(void)
{
extern uchar _binary_entryother_start[], _binary_entryother_size[];
uchar *code;
struct cpu *c;
char *stack;
// Write entry code to unused memory at 0x7000.
// The linker has placed the image of entryother.S in
// _binary_entryother_start.
code = p2v(0x7000);
memmove(code, _binary_entryother_start, (uint)_binary_entryother_size);
```
- entryother.S 的入口被 linked 到 `0x7000`,這裡將 code 指向 `0x7000` 作為 entryother.S 的進入點。
```c=+
for(c = cpus; c < cpus+ncpu; c++){
if(c == cpus+cpunum()) // We've started already.
continue;
// Tell entryother.S what stack to use, where to enter, and what
// pgdir to use. We cannot use kpgdir yet, because the AP processor
// is running in low memory, so we use entrypgdir for the APs too.
stack = kalloc();
*(void**)(code-4) = stack + KSTACKSIZE;
*(void**)(code-8) = mpenter;
*(int**)(code-12) = (void *) v2p(entrypgdir);
```
- 為待會的 entryother 建立一個堆疊 ...
```c=+
lapicstartap(c->id, v2p(code));
```
- 正式的啟動 CPU c,即進入 entryother.S
- entryother.S 做完設定後會 call mpenter(),mpmenter 最後會呼叫 mpmain()。
```c=+
// wait for cpu to finish mpmain()
while(c->started == 0)
;
}
}
```
- 在 mpmain() 會將 cpu->started 設為 1,CPU0 在 `while` 迴圈等待 CPU c 啟動完畢,才繼續啟動下一個 CPU。
### `mpenter()`
```c=46
static void
mpenter(void)
{
switchkvm();
seginit();
lapicinit();
mpmain();
}
```
### `mpmain()`
```c=56
static void
mpmain(void)
{
cprintf("cpu%d: starting\n", cpu->id);
idtinit(); // load idt register
xchg(&cpu->started, 1); // tell startothers() we're up
scheduler(); // start running processes
}
```
---
:::success
**File:** lapic.c
:::
### `lapicstartup()`
```c=137
void
lapicstartap(uchar apicid, uint addr)
{
int i;
ushort *wrv;
// "The BSP must initialize CMOS shutdown code to 0AH
// and the warm reset vector (DWORD based at 40:67) to point at
// the AP startup code prior to the [universal startup algorithm]."
outb(IO_RTC, 0xF); // offset 0xF is shutdown code
outb(IO_RTC+1, 0x0A);
wrv = (ushort*)P2V((0x40<<4 | 0x67)); // Warm reset vector
wrv[0] = 0;
wrv[1] = addr >> 4;
// "Universal startup algorithm."
// Send INIT (level-triggered) interrupt to reset other CPU.
lapicw(ICRHI, apicid<<24);
lapicw(ICRLO, INIT | LEVEL | ASSERT);
microdelay(200);
lapicw(ICRLO, INIT | LEVEL);
microdelay(100); // should be 10ms, but too slow in Bochs!
// Send startup IPI (twice!) to enter code.
// Regular hardware is supposed to only accept a STARTUP
// when it is in the halted state due to an INIT. So the second
// should be ignored, but it is part of the official Intel algorithm.
// Bochs complains about the second one. Too bad for Bochs.
for(i = 0; i < 2; i++){
lapicw(ICRHI, apicid<<24);
lapicw(ICRLO, STARTUP | (addr>>12));
microdelay(200);
}
}
```
---
## Main 解析
### `main()`
```c=17
int
main(void)
{
kinit1(end, P2V(4*1024*1024)); // phys page allocator
kvmalloc(); // kernel page table
mpinit(); // collect info about this machine
lapicinit();
seginit(); // set up segments
cprintf("\ncpu%d: starting xv6\n\n", cpu->id);
picinit(); // interrupt controller
ioapicinit(); // another interrupt controller
consoleinit(); // I/O devices & their interrupts
uartinit(); // serial port
pinit(); // process table
tvinit(); // trap vectors
binit(); // buffer cache
fileinit(); // file table
iinit(); // inode cache
ideinit(); // disk
if(!ismp)
timerinit(); // uniprocessor timer
startothers(); // start other processors
kinit2(P2V(4*1024*1024), P2V(PHYSTOP)); // must come after startothers()
userinit(); // first user process
// Finish setting up this processor in mpmain.
mpmain();
}
```
- `kinit1()`
- `kvmalloc()`
- `mpinit()`
- `lapicinit()`
- `seginit()`
- `picinit()`
- `ioapicinit()`
- `consoleinit()`
- `uartinit()`
- `pinit()`
- [`tvinit()`](https://hackmd.io/s/SyeO-X_4Q#tvinit)
- `binit()`
- `fileinit()`
- `iinit()`
- [`ideinit()`](https://hackmd.io/s/SyeO-X_4Q#ideinit)
- `timerinit()`
- [`startothers()`](https://hackmd.io/s/HkETnChUm#Code-startothers)
- `kinit2()`
- [`userinit()`](https://hackmd.io/s/BJjmDiZmX#Code-建立第一個-process)
- `mpmain()`
###### tags: `xv6` `kernel`