| #ifndef DEVICE_H |
| #define DEVICE_H |
| |
| #include <device/resource.h> |
| |
| struct device; |
| typedef struct device * device_t; |
| |
| struct device_operations { |
| void (*read_resources)(device_t dev); |
| void (*set_resources)(device_t dev); |
| void (*init)(device_t dev); |
| unsigned int (*scan_bus)(device_t bus, unsigned int max); |
| void (*enable)(device_t dev); |
| }; |
| |
| |
| #define MAX_RESOURCES 6 |
| /* |
| * There is one device structure for each slot-number/function-number |
| * combination: |
| */ |
| |
| struct device { |
| device_t bus; /* bus this device is on */ |
| device_t children; /* devices behind this bridge */ |
| device_t sibling; /* next device on this bus */ |
| device_t next; /* chain of all devices */ |
| |
| unsigned int devfn; /* encoded device & function index */ |
| unsigned short vendor; |
| unsigned short device; |
| unsigned int class; /* 3 bytes: (base,sub,prog-if) */ |
| unsigned int hdr_type; /* PCI header type */ |
| unsigned int enable : 1; /* set if we should enable the device */ |
| |
| unsigned char secondary; /* secondary bus number */ |
| unsigned char subordinate; /* max subordinate bus number */ |
| uint8_t command; |
| /* |
| * In theory, the irq level can be read from configuration |
| * space and all would be fine. However, old PCI chips don't |
| * support these registers and return 0 instead. For example, |
| * the Vision864-P rev 0 chip can uses INTA, but returns 0 in |
| * the interrupt line and pin registers. pci_init() |
| * initializes this field with the value at PCI_INTERRUPT_LINE |
| * and it is the job of pcibios_fixup() to change it if |
| * necessary. The field must not be 0 unless the device |
| * cannot generate interrupts at all. |
| */ |
| unsigned int irq; /* irq generated by this device */ |
| |
| /* Base registers for this device, can be adjusted by |
| * pcibios_fixup() as necessary. |
| */ |
| struct resource resource[MAX_RESOURCES]; |
| unsigned int resources; |
| unsigned long rom_address; |
| struct device_operations *ops; |
| }; |
| |
| extern struct device dev_root; /* root bus */ |
| extern struct device *all_devices; /* list of all devices */ |
| |
| |
| /* Generic device interface functions */ |
| extern void dev_enumerate(void); |
| extern void dev_configure(void); |
| extern void dev_enable(void); |
| extern void dev_initialize(void); |
| |
| /* Generic device helper functions */ |
| void append_device(device_t dev); |
| void compute_allocate_resource(device_t bus, struct resource *bridge, |
| unsigned long type_mask, unsigned long type); |
| void assign_resources(device_t bus); |
| void enumerate_static_device(void); |
| |
| /* Helper functions */ |
| device_t dev_find_device (unsigned int vendor, unsigned int device, device_t from); |
| device_t dev_find_class (unsigned int class, device_t from); |
| device_t dev_find_slot (unsigned int bus, unsigned int devfn); |
| |
| /* Rounding for boundaries. |
| * Due to some chip bugs, go ahead and roung IO to 16 |
| */ |
| #define DEVICE_IO_ALIGN 16 |
| #define DEVICE_MEM_ALIGN 4096 |
| |
| |
| #endif /* DEVICE_H */ |