#include <sys/types.h>
#include <sys/sunddi.h>
#include <sys/pci_impl.h>
#include <sys/ddi_subrdefs.h>
#include <sys/bootconf.h>
#include <sys/psw.h>
#include <sys/modctl.h>
#include <sys/errno.h>
#include <sys/pci.h>
#include <sys/pci_cfgspace.h>
#include <sys/reboot.h>
#include <sys/pci_cfgspace_impl.h>
#include <sys/mutex.h>
#include <sys/plat/pci_prd.h>
extern int pci_boot_debug;
extern int pci_boot_maxbus;
void pci_enumerate(int);
void pci_setup_tree(void);
void pci_reprogram(void);
dev_info_t *pci_boot_bus_to_dip(uint32_t);
static struct modlmisc modlmisc = {
&mod_miscops, "PCI BIOS interface"
};
static struct modlinkage modlinkage = {
MODREV_1, (void *)&modlmisc, NULL
};
static pci_prd_upcalls_t pci_upcalls = {
.pru_bus2dip_f = pci_boot_bus_to_dip
};
int
_init(void)
{
int err;
if ((err = pci_prd_init(&pci_upcalls)) != 0) {
return (err);
}
if ((err = mod_install(&modlinkage)) != 0) {
pci_prd_fini();
return (err);
}
impl_bus_add_probe(pci_enumerate);
return (0);
}
int
_fini(void)
{
int err;
if ((err = mod_remove(&modlinkage)) != 0)
return (err);
impl_bus_delete_probe(pci_enumerate);
pci_prd_fini();
return (0);
}
int
_info(struct modinfo *modinfop)
{
return (mod_info(&modlinkage, modinfop));
}
void
pci_enumerate(int reprogram)
{
extern void add_pci_fixes(void);
extern void undo_pci_fixes(void);
if (reprogram == 0) {
pci_boot_maxbus = pci_prd_max_bus();
}
add_pci_fixes();
if (reprogram) {
pci_reprogram();
undo_pci_fixes();
return;
}
pci_setup_tree();
undo_pci_fixes();
}