#include <sys/types.h>
#include <sys/atomic.h>
#include <sys/sunddi.h>
#include <sys/sunndi.h>
#include <sys/acpi/acpi.h>
#include <sys/acpica.h>
#include <sys/acpidev.h>
#include <sys/acpidev_impl.h>
#include <sys/pci.h>
static ACPI_STATUS acpidev_device_probe(acpidev_walk_info_t *infop);
static acpidev_filter_result_t acpidev_device_filter(acpidev_walk_info_t *infop,
char *devname, int maxlen);
static acpidev_filter_result_t acpidev_device_filter_usb(acpidev_walk_info_t *,
ACPI_HANDLE, acpidev_filter_rule_t *, char *, int);
static ACPI_STATUS acpidev_device_init(acpidev_walk_info_t *infop);
static uint32_t acpidev_device_unitaddr = 0;
acpidev_class_t acpidev_class_device = {
0,
ACPIDEV_CLASS_REV1,
ACPIDEV_CLASS_ID_DEVICE,
"ACPI Device",
ACPIDEV_TYPE_DEVICE,
NULL,
NULL,
NULL,
acpidev_device_probe,
acpidev_device_filter,
acpidev_device_init,
NULL,
};
acpidev_class_list_t *acpidev_class_list_device = NULL;
static acpidev_filter_rule_t acpidev_device_filters[] = {
{
NULL,
0,
ACPIDEV_FILTER_DEFAULT,
&acpidev_class_list_device,
1,
1,
ACPIDEV_OBJECT_NAME_SB,
ACPIDEV_NODE_NAME_MODULE_SBD,
},
{
NULL,
0,
ACPIDEV_FILTER_SKIP,
NULL,
1,
1,
NULL,
NULL,
},
{
acpidev_device_filter_usb,
0,
ACPIDEV_FILTER_SCAN,
&acpidev_class_list_usbport,
2,
INT_MAX,
NULL,
NULL
},
{
NULL,
0,
ACPIDEV_FILTER_SCAN,
&acpidev_class_list_device,
2,
INT_MAX,
NULL,
NULL,
}
};
static ACPI_STATUS
acpidev_device_probe(acpidev_walk_info_t *infop)
{
ACPI_STATUS rc = AE_OK;
int flags;
ASSERT(infop != NULL);
ASSERT(infop->awi_hdl != NULL);
ASSERT(infop->awi_info != NULL);
if (infop->awi_info->Type != ACPI_TYPE_DEVICE) {
return (AE_OK);
}
flags = ACPIDEV_PROCESS_FLAG_SCAN;
switch (infop->awi_op_type) {
case ACPIDEV_OP_BOOT_PROBE:
flags |= ACPIDEV_PROCESS_FLAG_CREATE;
break;
case ACPIDEV_OP_BOOT_REPROBE:
break;
case ACPIDEV_OP_HOTPLUG_PROBE:
flags |= ACPIDEV_PROCESS_FLAG_CREATE |
ACPIDEV_PROCESS_FLAG_SYNCSTATUS |
ACPIDEV_PROCESS_FLAG_HOLDBRANCH;
break;
default:
ACPIDEV_DEBUG(CE_WARN,
"!acpidev: unknown operation type %u in "
"acpi_device_probe().", infop->awi_op_type);
rc = AE_BAD_PARAMETER;
break;
}
if (rc == AE_OK) {
rc = acpidev_process_object(infop, flags);
}
if (ACPI_FAILURE(rc) && rc != AE_NOT_EXIST && rc != AE_ALREADY_EXISTS) {
cmn_err(CE_WARN,
"!acpidev: failed to process device object %s.",
infop->awi_name);
} else {
rc = AE_OK;
}
return (rc);
}
static acpidev_filter_result_t
acpidev_device_filter_usb(acpidev_walk_info_t *infop, ACPI_HANDLE hdl,
acpidev_filter_rule_t *afrp, char *devname, int len)
{
dev_info_t *dip;
char **compat;
uint_t ncompat, i;
if (infop->awi_op_type != ACPIDEV_OP_BOOT_REPROBE)
return (ACPIDEV_FILTER_SKIP);
if (ACPI_FAILURE(acpica_get_devinfo(hdl, &dip))) {
return (ACPIDEV_FILTER_SKIP);
}
if (ddi_prop_lookup_string_array(DDI_DEV_T_ANY, dip, DDI_PROP_DONTPASS,
"compatible", &compat, &ncompat) != DDI_SUCCESS) {
return (ACPIDEV_FILTER_SKIP);
}
for (i = 0; i < ncompat; i++) {
if (strcmp(compat[i], "pciclass,0c03") == 0 ||
strcmp(compat[i], "pciexclass,0c03") == 0) {
ddi_prop_free(compat);
return (ACPIDEV_FILTER_SCAN);
}
}
ddi_prop_free(compat);
return (ACPIDEV_FILTER_SKIP);
}
static acpidev_filter_result_t
acpidev_device_filter(acpidev_walk_info_t *infop, char *devname, int maxlen)
{
acpidev_filter_result_t res;
ASSERT(infop != NULL);
if (infop->awi_op_type == ACPIDEV_OP_BOOT_PROBE ||
infop->awi_op_type == ACPIDEV_OP_BOOT_REPROBE ||
infop->awi_op_type == ACPIDEV_OP_HOTPLUG_PROBE) {
res = acpidev_filter_device(infop, infop->awi_hdl,
ACPIDEV_ARRAY_PARAM(acpidev_device_filters),
devname, maxlen);
} else {
ACPIDEV_DEBUG(CE_WARN, "!acpidev: unknown operation type %u "
"in acpidev_device_filter().", infop->awi_op_type);
res = ACPIDEV_FILTER_FAILED;
}
return (res);
}
static ACPI_STATUS
acpidev_device_init(acpidev_walk_info_t *infop)
{
char unitaddr[32];
char *compatible[] = {
ACPIDEV_TYPE_DEVICE,
ACPIDEV_HID_VIRTNEX,
ACPIDEV_TYPE_VIRTNEX,
};
if (ACPI_FAILURE(acpidev_set_compatible(infop,
ACPIDEV_ARRAY_PARAM(compatible)))) {
return (AE_ERROR);
}
(void) snprintf(unitaddr, sizeof (unitaddr), "%u",
atomic_inc_32_nv(&acpidev_device_unitaddr) - 1);
if (ACPI_FAILURE(acpidev_set_unitaddr(infop, NULL, 0, unitaddr))) {
return (AE_ERROR);
}
return (AE_OK);
}