#include <sys/types.h>
#include <sys/atomic.h>
#include <sys/note.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_dr.h>
#include <sys/acpidev_impl.h>
static ACPI_STATUS acpidev_container_probe(acpidev_walk_info_t *infop);
static acpidev_filter_result_t acpidev_container_filter(
acpidev_walk_info_t *infop, char *devname, int maxlen);
static ACPI_STATUS acpidev_container_init(acpidev_walk_info_t *infop);
static acpidev_filter_result_t acpidev_container_filter_func(
acpidev_walk_info_t *infop, ACPI_HANDLE hdl, acpidev_filter_rule_t *rulep,
char *devname, int devnamelen);
acpidev_class_t acpidev_class_container = {
0,
ACPIDEV_CLASS_REV1,
ACPIDEV_CLASS_ID_CONTAINER,
"ACPI Container",
ACPIDEV_TYPE_CONTAINER,
NULL,
NULL,
NULL,
acpidev_container_probe,
acpidev_container_filter,
acpidev_container_init,
NULL,
};
static char *acpidev_container_device_ids[] = {
ACPIDEV_HID_MODULE,
ACPIDEV_HID_CONTAINER1,
ACPIDEV_HID_CONTAINER2,
};
static char *acpidev_container_uid_formats[] = {
"CPUSCK%x",
};
static acpidev_filter_rule_t acpidev_container_filters[] = {
{
NULL,
0,
ACPIDEV_FILTER_SKIP,
NULL,
1,
1,
NULL,
NULL,
},
{
acpidev_container_filter_func,
0,
ACPIDEV_FILTER_DEFAULT,
&acpidev_class_list_device,
2,
INT_MAX,
NULL,
ACPIDEV_NODE_NAME_CONTAINER,
}
};
static ACPI_STATUS
acpidev_container_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 ||
acpidev_match_device_id(infop->awi_info,
ACPIDEV_ARRAY_PARAM(acpidev_container_device_ids)) == 0) {
return (AE_OK);
}
flags = ACPIDEV_PROCESS_FLAG_SCAN;
switch (infop->awi_op_type) {
case ACPIDEV_OP_BOOT_PROBE:
if (acpica_get_devcfg_feature(ACPI_DEVCFG_CONTAINER)) {
flags |= ACPIDEV_PROCESS_FLAG_CREATE;
acpidev_dr_check(infop);
}
break;
case ACPIDEV_OP_BOOT_REPROBE:
break;
case ACPIDEV_OP_HOTPLUG_PROBE:
if (acpica_get_devcfg_feature(ACPI_DEVCFG_CONTAINER)) {
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 "
"acpidev_container_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 container object %s.",
infop->awi_name);
} else {
rc = AE_OK;
}
return (rc);
}
static ACPI_STATUS
acpidev_container_search_dev(ACPI_HANDLE hdl, UINT32 lvl, void *ctx,
void **retval)
{
_NOTE(ARGUNUSED(hdl, retval));
int *fp = (int *)ctx;
*fp = lvl;
return (AE_CTRL_TERMINATE);
}
static acpidev_filter_result_t
acpidev_container_filter_func(acpidev_walk_info_t *infop, ACPI_HANDLE hdl,
acpidev_filter_rule_t *rulep, char *devname, int devnamelen)
{
ACPI_BUFFER buf;
void *retval;
int proc_lvl, cpu_lvl, module_lvl;
acpidev_filter_result_t res;
static char *cpu_hids[] = {
ACPIDEV_HID_CPU,
};
static char *module_hids[] = {
ACPIDEV_HID_MODULE,
};
res = acpidev_filter_default(infop, hdl, rulep, devname, devnamelen);
if (devname == NULL || res == ACPIDEV_FILTER_FAILED ||
res == ACPIDEV_FILTER_SKIP) {
return (res);
}
retval = NULL;
proc_lvl = INT_MAX;
cpu_lvl = INT_MAX;
module_lvl = INT_MAX;
(void) AcpiWalkNamespace(ACPI_TYPE_PROCESSOR, hdl, 2,
acpidev_container_search_dev, NULL, &proc_lvl, &retval);
(void) acpidev_get_device_by_id(hdl, ACPIDEV_ARRAY_PARAM(cpu_hids), 2,
B_FALSE, acpidev_container_search_dev, &cpu_lvl, &retval);
(void) acpidev_get_device_by_id(hdl, ACPIDEV_ARRAY_PARAM(module_hids),
2, B_FALSE, acpidev_container_search_dev, &module_lvl, &retval);
buf.Pointer = devname;
buf.Length = devnamelen;
if (cpu_lvl > proc_lvl) {
cpu_lvl = proc_lvl;
}
if (cpu_lvl == 1) {
(void) strlcpy(devname, ACPIDEV_NODE_NAME_MODULE_CPU,
devnamelen);
} else if (cpu_lvl == 2 && module_lvl == 1) {
(void) strlcpy(devname, ACPIDEV_NODE_NAME_MODULE_SBD,
devnamelen);
} else if (ACPI_FAILURE(AcpiGetName(infop->awi_hdl,
ACPI_SINGLE_NAME, &buf))) {
(void) strlcpy(devname, ACPIDEV_NODE_NAME_CONTAINER,
devnamelen);
}
return (res);
}
static acpidev_filter_result_t
acpidev_container_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_container_filters),
devname, maxlen);
} else {
res = ACPIDEV_FILTER_FAILED;
}
return (res);
}
static ACPI_STATUS
acpidev_container_init(acpidev_walk_info_t *infop)
{
static char *compatible[] = {
ACPIDEV_TYPE_CONTAINER,
ACPIDEV_HID_VIRTNEX,
ACPIDEV_TYPE_VIRTNEX,
};
ASSERT(infop != NULL);
ASSERT(infop->awi_hdl != NULL);
ASSERT(infop->awi_dip != NULL);
if (ACPI_FAILURE(acpidev_set_compatible(infop,
ACPIDEV_ARRAY_PARAM(compatible)))) {
return (AE_ERROR);
}
if (ACPI_FAILURE(acpidev_set_unitaddr(infop,
ACPIDEV_ARRAY_PARAM(acpidev_container_uid_formats), NULL))) {
return (AE_ERROR);
}
return (AE_OK);
}