vbus->addr = VIRTIO_BASE_ADDR;
// 初始化设备
/\* virtio console \*/
if (p->console) {
s->common.console_dev = virtio\_console\_init(vbus, p->console);
vbus->addr += VIRTIO_SIZE;
}
...
if (p->input_device) {
// 键盘
s->keyboard_dev = virtio\_input\_init(vbus,
VIRTIO_INPUT_TYPE_KEYBOARD);
vbus->addr += VIRTIO_SIZE;
// 鼠标
s->mouse_dev = virtio\_input\_init(vbus,
VIRTIO_INPUT_TYPE_TABLET);
vbus->addr += VIRTIO_SIZE;
}
// 拷贝BIOS和Kernel;手动写入5条指令
copy\_bios(s, p->files[VM_FILE_BIOS].buf, p->files[VM_FILE_BIOS].len,
p->files[VM_FILE_KERNEL].buf, p->files[VM_FILE_KERNEL].len,
p->files[VM_FILE_INITRD].buf, p->files[VM_FILE_INITRD].len,
p->cmdline);
return (VirtMachine \*)s;
}
首先,初始化VirtMachineClass、ram大小、max\_xlen,以及内存映射初始化等。
然后,在riscv\_cpu\_init函数中,会完成pc赋初值和TLB初始化(赋值为-1)。
s->pc = 0x1000;
s->cpu_state = riscv_cpu_init(s->mem_map, max_xlen);
>
> cpu\_state的类型为RISCVCPUState结构,该结构中,包含mstatus、mtvec、mscratch等CSR寄存器定义。
>
>
>
**我们猜测,第一条指令地址,就是0x1000**。
初始化结构参数,其实就是把一些参数,保存到RISCVMachine对象中。
## 2 配置RAM地址空间
我们对本部分代码,进行分析,并结合以下