big kernel additions: paging, elf loading, separate kernel and user page allocation it now properly loads and runs sd0:bin/init.elf still need to determine which disk was booted from, and start the init on that disk
75 lines
No EOL
2 KiB
C
75 lines
No EOL
2 KiB
C
#include "panic.h"
|
|
#include "boot.h"
|
|
#include "util.h"
|
|
#include "pmap.h"
|
|
#include "pci.h"
|
|
|
|
enum {
|
|
PCP_SELECT = 0x0cf8,
|
|
PCP_READ = 0x0cfc
|
|
};
|
|
|
|
uint16_t n_pci_devices;
|
|
static struct pci_device *pci_device_pages[256];
|
|
|
|
#define PCI_DEVICES_PER_PAGE (4096 / sizeof(struct pci_device))
|
|
|
|
struct pci_device *nth_pci_device(uint16_t n) {
|
|
return pci_device_pages[n / PCI_DEVICES_PER_PAGE] + (n % PCI_DEVICES_PER_PAGE);
|
|
}
|
|
|
|
struct pci_device *find_pci_device_from_class_and_subclass(uint8_t class, uint8_t subclass, uint16_t start, uint16_t *index) {
|
|
for (uint16_t n = start; n < n_pci_devices; ++n) {
|
|
struct pci_device *p = nth_pci_device(n);
|
|
if ((p->class == class) && (p->subclass == subclass)) {
|
|
*index = n;
|
|
return p;
|
|
}
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
static struct pci_device *next_pci_device() {
|
|
if (!(n_pci_devices % PCI_DEVICES_PER_PAGE))
|
|
if (!(pci_device_pages[n_pci_devices / PCI_DEVICES_PER_PAGE] = allocate_kernel_pages(1)))
|
|
panic("Out of memory in PCI enumeration");
|
|
return nth_pci_device(n_pci_devices++);
|
|
}
|
|
|
|
static uint32_t pci_read_config(uint16_t number, uint8_t reg) {
|
|
uint32_t cspace_addr = 0x80000000 | (number << 8) | (reg << 2);
|
|
outd(PCP_SELECT, cspace_addr);
|
|
|
|
return ind(PCP_READ);
|
|
}
|
|
|
|
static void pci_device_check(uint16_t number) {
|
|
uint32_t id = pci_read_config(number, 0);
|
|
if ((id & 0xffff) == 0xffff)
|
|
return;
|
|
|
|
struct pci_device *next_device = next_pci_device();
|
|
|
|
next_device->number = number;
|
|
|
|
next_device->id_vendor = id;
|
|
next_device->id_device = id >> 16;
|
|
|
|
uint32_t class = pci_read_config(number, 2);
|
|
|
|
next_device->class = class >> 24;
|
|
next_device->subclass = class >> 16;
|
|
next_device->iface = class >> 8;
|
|
}
|
|
|
|
void pci_init() {
|
|
if (!(BOOT_INFO->support_flags & BIS_PCI))
|
|
panic("No PCI support detected.");
|
|
if (!(BOOT_INFO->pci_hw_char & PHC_CS_M1))
|
|
panic("No PCI Mechanism 1 support");
|
|
|
|
n_pci_devices = 0;
|
|
|
|
for (uint32_t number = 0; number < (BOOT_INFO->last_pci_bus + 1) * 256; ++number)
|
|
pci_device_check(number);
|
|
} |