summaryrefslogtreecommitdiff
path: root/src/kernel/pci.c
blob: 28ca6c68a145f81908461c64f7f85309648c4e9b (plain) (blame)
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
#include "panic.h"
#include "boot.h"
#include "util.h"
#include "mem.h"
#include "pci.h"

enum {
  PCP_SELECT = 0x0cf8,
  PCP_READ   = 0x0cfc
};

uint16_t n_pci_devices = 0;
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_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");

  for (uint32_t number = 0; number < (BOOT_INFO->last_pci_bus + 1) * 256; ++number)
    pci_device_check(number);
}