Chameleon Applications

Chameleon Applications Svn Source Tree

Root/branches/iFabio/Chameleon/i386/libsaio/pci.c

Source at commit 307 created 12 years 11 months ago.
By ifabio, merge changes from trunk (929). Also merge the module changes from Azimutz branche (fix compile error) Also edited the info.plist into AHCIPortInjector.kext: http://forum.voodooprojects.org/index.php/topic,1170.0.html
1/*
2 *
3 * Copyright 2008 by Islam M. Ahmed Zaid. All rights reserved.
4 *
5 */
6
7#include "libsaio.h"
8#include "bootstruct.h"
9#include "pci.h"
10#include "pci_root.h"
11
12#ifndef DEBUG_PCI
13#define DEBUG_PCI 0
14#endif
15
16#if DEBUG_PCI
17#define DBG(x...)printf(x)
18#else
19#define DBG(x...)
20#endif
21
22pci_dt_t*root_pci_dev;
23
24
25uint8_t pci_config_read8(uint32_t pci_addr, uint8_t reg)
26{
27pci_addr |= reg & ~3;
28outl(PCI_ADDR_REG, pci_addr);
29return inb(PCI_DATA_REG + (reg & 3));
30}
31
32uint16_t pci_config_read16(uint32_t pci_addr, uint8_t reg)
33{
34pci_addr |= reg & ~3;
35outl(PCI_ADDR_REG, pci_addr);
36return inw(PCI_DATA_REG + (reg & 2));
37}
38
39uint32_t pci_config_read32(uint32_t pci_addr, uint8_t reg)
40{
41pci_addr |= reg & ~3;
42outl(PCI_ADDR_REG, pci_addr);
43return inl(PCI_DATA_REG);
44}
45
46void pci_config_write8(uint32_t pci_addr, uint8_t reg, uint8_t data)
47{
48pci_addr |= reg & ~3;
49outl(PCI_ADDR_REG, pci_addr);
50outb(PCI_DATA_REG + (reg & 3), data);
51}
52
53void pci_config_write16(uint32_t pci_addr, uint8_t reg, uint16_t data)
54{
55pci_addr |= reg & ~3;
56outl(PCI_ADDR_REG, pci_addr);
57outw(PCI_DATA_REG + (reg & 2), data);
58}
59
60void pci_config_write32(uint32_t pci_addr, uint8_t reg, uint32_t data)
61{
62pci_addr |= reg & ~3;
63outl(PCI_ADDR_REG, pci_addr);
64outl(PCI_DATA_REG, data);
65}
66
67void scan_pci_bus(pci_dt_t *start, uint8_t bus)
68{
69pci_dt_t*new;
70pci_dt_t**current = &start->children;
71uint32_tid;
72uint32_tpci_addr;
73uint8_tdev;
74uint8_tfunc;
75uint8_tsecondary_bus;
76uint8_theader_type;
77
78for (dev = 0; dev < 32; dev++) {
79for (func = 0; func < 8; func++) {
80pci_addr = PCIADDR(bus, dev, func);
81id = pci_config_read32(pci_addr, PCI_VENDOR_ID);
82if (!id || id == 0xffffffff) {
83continue;
84}
85new = (pci_dt_t*)malloc(sizeof(pci_dt_t));
86bzero(new, sizeof(pci_dt_t));
87new->dev.addr= pci_addr;
88new->vendor_id= id & 0xffff;
89new->device_id= (id >> 16) & 0xffff;
90new->subsys_id.subsys_id= pci_config_read32(pci_addr, PCI_SUBSYSTEM_VENDOR_ID);
91new->class_id= pci_config_read16(pci_addr, PCI_CLASS_DEVICE);
92new->parent= start;
93
94header_type = pci_config_read8(pci_addr, PCI_HEADER_TYPE);
95switch (header_type & 0x7f) {
96case PCI_HEADER_TYPE_BRIDGE:
97case PCI_HEADER_TYPE_CARDBUS:
98secondary_bus = pci_config_read8(pci_addr, PCI_SECONDARY_BUS);
99if (secondary_bus != 0) {
100scan_pci_bus(new, secondary_bus);
101}
102break;
103}
104*current = new;
105current = &new->next;
106
107if ((func == 0) && ((header_type & 0x80) == 0)) {
108break;
109}
110}
111}
112}
113
114void enable_pci_devs(void)
115{
116uint16_t id;
117uint32_t rcba, *fd;
118
119id = pci_config_read16(PCIADDR(0, 0x00, 0), 0x00);
120/* make sure we're on Intel chipset */
121if (id != 0x8086)
122return;
123rcba = pci_config_read32(PCIADDR(0, 0x1f, 0), 0xf0) & ~1;
124fd = (uint32_t *)(rcba + 0x3418);
125/* set SMBus Disable (SD) to 0 */
126*fd &= ~0x8;
127/* and all devices? */
128//*fd = 0x1;
129}
130
131
132void build_pci_dt(void)
133{
134root_pci_dev = malloc(sizeof(pci_dt_t));
135bzero(root_pci_dev, sizeof(pci_dt_t));
136enable_pci_devs();
137scan_pci_bus(root_pci_dev, 0);
138
139#if DEBUG_PCI
140dump_pci_dt(root_pci_dev->children);
141pause();
142#endif
143}
144
145static char dev_path[256];
146char *get_pci_dev_path(pci_dt_t *pci_dt)
147{
148pci_dt_t*current;
149pci_dt_t*end;
150chartmp[64];
151
152dev_path[0] = 0;
153end = root_pci_dev;
154
155int uid = getPciRootUID();
156while (end != pci_dt)
157{
158current = pci_dt;
159while (current->parent != end)
160current = current->parent;
161end = current;
162if (current->parent == root_pci_dev)
163{
164sprintf(tmp, "PciRoot(0x%x)/Pci(0x%x,0x%x)", uid,
165current->dev.bits.dev, current->dev.bits.func);
166} else {
167sprintf(tmp, "/Pci(0x%x,0x%x)",
168current->dev.bits.dev, current->dev.bits.func);
169}
170strcat(dev_path, tmp);
171}
172return dev_path;
173}
174
175void dump_pci_dt(pci_dt_t *pci_dt)
176{
177pci_dt_t*current;
178
179current = pci_dt;
180while (current) {
181printf("%02x:%02x.%x [%04x] [%04x:%04x] (subsys [%04x:%04x]):: %s\n",
182current->dev.bits.bus, current->dev.bits.dev, current->dev.bits.func,
183current->class_id, current->vendor_id, current->device_id,
184current->subsys_id.subsys.vendor_id, current->subsys_id.subsys.device_id,
185get_pci_dev_path(current));
186dump_pci_dt(current->children);
187current = current->next;
188}
189}
190

Archive Download this file

Revision: 307